理解javaScript变量、做用域链

变量
因为JavaScript变量松散类型的本质,决定了它只是在特定时间用于保存特定值的一个名字而已。变量分为基础类型值引用类型值
基本类型值指的是简单的数据段(Undefined、Null、Boolean、Number、String 和 symbol),而引用类型值指那些可能由多个值构成的对象。前端

引用类型的值是保存在内存中的对象。与其余语言不一样,JavaScript 不容许直接访问内存中的位置, 也就是说不能直接操做对象的内存空间。在操做对象时,其实是在操做对象的引用而不是实际的对象。 为此,引用类型的值是按引用访问的
var obj1 = new Object();web

var obj2 = obj1;
obj1.name = "Nicholas";
alert(obj2.name);  //"Nicholas"

clipboard.png

做用域链
1.执行环境
执行环境(execution context,为简单起见,有时也称为“环境”)定义了变量或函数有权访问的其余数据,决定了它们各自的行为。
每一个函数都有本身的执行环境。当执行流进入一个函数时,函数的环境就会被推入一个环境栈中。 而在函数执行以后,栈将其环境弹出,把控制权返回给以前的执行环境。ECMAScript 程序中的执行流 正是由这个方便的机制控制着。
全局执行环境是最外层环境,web开发中一般认为是window;某个执行环境中全部代码执行完毕后,该环境将被销毁,保存在其中的变量和函数也将同时被销毁
2.当代码在一个环境中执行时,会建立变量对象的一个做用域链(scope chain)。做用域链的用途,是保证对执行环境有权访问的全部变量和函数的有序访问。当前的执行环境就是做用域的最前端,标识符解析是沿着做用域链最前端向后回溯,直到找到标志符
3.当某个函数被调用时,会建立一个执行环境(execution context)及相应的做用域链。 而后,使用 arguments 和其余命名参数的值来初始化函数的活动对象(activation object)。
闭包
闭包是指有权访问另外一个函数做用域中的变量的函数。建立闭包的常见方式,就是在一个函数内部建立另外一个函数。
不管何时在函数中访问一个变量时,就会从做用域链中搜索具备相应名字的变量。通常来说, 当函数执行完毕后,局部活动对象就会被销毁,内存中仅保存全局做用域(全局执行环境的变量对象)。 可是,闭包的状况又有所不一样。
外部函数执行结束后,执行环境就会在做用域链中销毁,可是因为闭包函数的局部做用域链仍在使用其活动对象,因此闭包函数被销毁时外部活动环境才能被销毁。
因为闭包会携带包含它的函数的做用域,所以会比其余函数占用更多的内存。闭包

this
1.(默认绑定)普通函数调用;this指向windowapp

clipboard.png

在严格模式下(strict mode),全局对象将没法使用默认绑定,即执行会报undefined的错误函数

2.(隐式绑定)做为对象方法调用;this指向调用对象this

clipboard.png

3.做为构造函数调用,this 指代实例对象spa

clipboard.png

4.call() 和 apply()、bind()
主要是经过改变对象的prototype关联对象。具体使用上,能够经过这两个方法call(…)或apply(…)来实现(大多数函数及本身建立的函数默认都提供这两个方法)call与apply是一样的做用,区别只是其余参数的设置上,
apply:调用一个对象的一个方法,用另外一个对象替换当前对象。例如:B.apply(A, arguments);即A对象应用B对象的方法。prototype

call:调用一个对象的一个方法,用另外一个对象替换当前对象。例如:B.call(A, args1,args2);即A对象调用B对象的方法。code

bind:call和apply都是改变上下文中的this并当即执行这个函数,bind方法可让对应的函数想何时调就何时调用,而且能够将参数在执行的时候添加,这是它们的区别。
bind也能够有多个参数,而且参数能够执行的时候再次添加,可是要注意的是,参数是按照形参的顺序进行的。对象

clipboard.png

相关文章
相关标签/搜索