函数声明和函数表达式

声明和赋值

js 复制代码
var myName = 'Lily'

上面这段代码可以看成是两行代码组成的:

js 复制代码
var myName //声明部分
myName = 'Lily' // 赋值部分

函数声明

js 复制代码
function foo() {
    console.log('foo')
}

函数foo是一个完整的函数声明,没有涉及到赋值操作

函数表达式

首先,分析下面这段代码

js 复制代码
var bar = function () { 
    console.log('bar')
 }

当执行这段代码的时候,V8在编译阶段会先查找声明语句,你可以把这段代码拆分为下面两行代码

js 复制代码
var bar = undefined
bar = function () {
    console.log('bar')
}

第一行是声明语句,所以V8在解析阶段,就会在作用域中创建该对象,并且将该对象设置为undefined,第二行是赋值/表达式,在编译阶段,V8并不会处理函数表达式,所以也就不会将该函数表达式提升到作用域中了。

变量提升

所谓的变量提升,是指在JavaScript代码执行过程中,JavaScrpt引擎把变量的声明部分和函数的声明部分提升到代码开头的"行为"。

对于变量提升,如果是一个普通变量,变量提升之后都是undefined,如果是声明的函数,那么变量提升之后的值则是函数对象。

在编译阶段,如果解析到函数声明, 那么V8会将这个函数声明转换为内存中的函数对象,并将其放在作用域中。

JavaScript执行流程

  • 首先JavaScript引擎会将代码中声明的变量和函数保存到执行上下文的变量环境中(编译阶段);
  • 然后JavaScript引擎会把声明之外的代码编译为字节码(编译阶段);

输入一段代码,经过编译后,会生成两部分内容:执行上下文(Execution context)和可执行代码。
执行上下文是JavaScript执行一段代码时的运行环境。

  • 最后JavaScript引擎执行"可执行代码"(执行阶段)。

测试题

js 复制代码
var fun = function (x, y) {
    return x - y 
}

function fun (x, y) {
    return x + y
}

console.log(fun(4, 3))

这道题目的最终输出是:1

执行过程分析:

  • 编译阶段,编译第一行代码时,会将变量fun的声明(var fun = undefined)保存在执行上下文的变量环境中;
  • 编译阶段,编译第五行代码时,会将函数fun声明保存在变量环境中,此时函数名fun与变量名fun重名,所以fun会被重新声明为函数对象。
  • 编译阶段,将声明之外的代码编译成字节码
  • 执行阶段,执行第一行代码时,将变量fun赋值为function(x, y) { return x -y }
  • 执行阶段,执行第五行时声明语句,应该不会编译为可执行代码
  • 执行阶段,调用fun函数,
  • 所以最终输出为1
相关推荐
程序员 沐阳43 分钟前
JavaScript 内存与引用:深究深浅拷贝、垃圾回收与 WeakMap/WeakSet
开发语言·javascript·ecmascript
cyclv3 小时前
无网络地图展示轨迹,地图瓦片下载,绘制管线
前端·javascript
HIT_Weston6 小时前
41、【Agent】【OpenCode】本地代理分析(五)
javascript·人工智能·opencode
前端Hardy6 小时前
前端必看!LocalStorage这么用,再也不踩坑(多框架通用,直接复制)
前端·javascript·面试
前端Hardy6 小时前
前端必看!前端路由守卫这么写,再也不担心权限混乱(Vue/React通用)
前端·javascript·面试
竹林8186 小时前
从ethers.js迁移到Viem:我在重构DeFi前端时踩过的那些坑
前端·javascript
前端郭德纲7 小时前
JavaScript Object.freeze() 详解
开发语言·javascript·ecmascript
希望永不加班7 小时前
SpringBoot 静态资源访问(图片/JS/CSS)配置详解
java·javascript·css·spring boot·后端
m0_738120728 小时前
渗透基础知识ctfshow——Web应用安全与防护(第一章)
服务器·前端·javascript·安全·web安全·网络安全
持续前行8 小时前
通过 npm 下载node_modules 某个依赖 ;例如 下载 @rollup/rollup-linux-arm64-gnu
前端·javascript·vue.js