JS是单线程的语言,执行顺序肯定是顺序执行,但是JS 引擎并不是一行一行地分析和执行程序,而是一段一段地分析执行,会先进行编译阶段然后才是执行阶段。
例子一:变量提升
foo;  // undefined
var foo = function () {
    console.log('foo1');
}
foo();  // foo1,foo赋值
var foo = function () {
    console.log('foo2');
}
foo(); // foo2,foo重新赋值例子二:函数提升
foo();  // foo2
function foo() {
    console.log('foo1');
}
foo();  // foo2
function foo() {
    console.log('foo2');
}
foo(); // foo2例子三:声明优先级,函数 > 变量
foo();  // foo2
var foo = function() {
    console.log('foo1');
}
foo();  // foo1,foo重新赋值
function foo() {
    console.log('foo2');
}
foo(); // foo1上面三个例子中,第一个例子是变量提升,第二个例子是函数提升,第三个例子是函数声明优先级高于变量声明。
需要注意的是同一作用域下存在多个同名函数声明,那么后面的会替换前面的函数声明。
执行上下文总共有三种类型
this 指向这个全局对象。eval 函数中的代码,很少用而且不建议使用。因为JS引擎创建了很多的执行上下文,所以JS引擎创建了执行上下文栈(Execution context stack,ECS)来管理执行上下文。
当 JavaScript 初始化的时候会向执行上下文栈压入一个全局执行上下文,我们用 globalContext 表示它,并且只有当整个应用程序结束的时候,执行栈才会被清空,所以程序结束之前, 执行栈最底部永远有个 globalContext。
ECStack = [     // 使用数组模拟栈
    globalContext
];有如下两段代码,执行的结果是一样的,但是两段代码究竟有什么不同?
//代码一
var scope = "global scope";
function checkscope(){
    var scope = "local scope";
    function f(){
        return scope;
    }
    return f();
}
checkscope();
//代码二
var scope = "global scope";
function checkscope(){
    var scope = "local scope";
    function f(){
        return scope;
    }
    return f;
}
checkscope()();答案是 执行上下文栈的变化不一样。
第一段代码:
ECStack.push(<checkscope> functionContext);
ECStack.push(<f> functionContext);//这里checkscope函数还没出栈,且函数f入栈
ECStack.pop();
ECStack.pop();第二段代码:
ECStack.push(<checkscope> functionContext);
ECStack.pop();//这里checkscope函数已经执行结束,出栈
ECStack.push(<f> functionContext);
ECStack.pop();在函数上下文中,用活动对象(activation object, AO)来表示变量对象。
活动对象和变量对象的区别在于
调用函数时,会为其创建一个Arguments对象,并自动初始化局部变量arguments,指代该Arguments对象。所有作为参数传入的值都会成为Arguments对象的数组元素。
执行上下文的代码会分成两个阶段进行处理
很明显,这个时候还没有执行代码
此时的变量对象会包括(如下顺序初始化):
看例子:
function foo(a) {
  var b = 2;
  function c() {}
  var d = function() {};
  b = 3;
}
foo(1);对于上面的代码,这个时候的AO是
AO = {
    arguments: {
        0: 1,
        length: 1
    },
    a: 1,
    b: undefined, 
    c: reference to function c(){},
    d: undefined
}形参arguments这时候已经有赋值了,但是变量还是undefined,只是初始化的值
这个阶段会顺序执行代码,修改变量对象的值,执行完成后AO如下:
AO = {
    arguments: {
        0: 1,
        length: 1
    },
    a: 1,
    b: 3,
    c: reference to function c(){},
    d: reference to FunctionExpression "d"
}在这里变量才会被赋值了。
总结如下:
4、在代码执行阶段,会再次修改变量对象的属性值
原文:https://www.cnblogs.com/w-yh/p/11967552.html