JS 中的 this 对每位前端工程师都不陌生,常常看到对象这里 this 那里 this,那什么是 this?答案就是上下文对象,即被调用函数所处的环境,也就是说,this 在函数内部指向了调用函数的对象。前端
通俗的讲,就是谁调用了函数。es6
this 指向 window数组
var name = 'xiaoming' // 思考,为何不能用 let 或者 const ? function foo () { console.log(this.name) } foo() // xiaoming
谁调用了这个函数,答案就是 window。这好理解,由于这里的变量和函数都是直接挂在 window 上的,等同于 window.foo()。需注意,严格模式下,this 为 undefined
前端工程师
this 指向一个对象app
var name = 'xiaoming' var foo = { name: 'Jon', getName () { console.log(this.name) } } foo.getName() // Jon
谁调用了这个函数,答案是 foo 对象,因此打印了 Jon 而不是 xiaoming函数
var bar = foo.getName bar() // xiaoming
若是赋值到另外一个变量,就变成 window 调用,因此打印了 xiaomingpost
this 指向了一个用 new 新生成的对象this
function Person (name) { this.name = name this.getName = function () { console.log(this.name) } } var jser = new Person('Jon') jser.getName() // Jon
这种方式成为 new 绑定,也叫作构造调用。spa
JavaScript 中,new 的机制实际上和面向类的语言彻底不一样,构造函数只是一些使用 new 操做符时被调用的函数,它们并不会属于某个类,也不会实例化一个类。prototype
实际上,除 ES6 的 Symbol()外,全部函数均可以用 new 来调用,因此并不存在所谓的构造函数,只有对于函数进行构造调用。
使用 new 来调用函数,或者说发生构造函数调用时,会自动执行下面的操做:
这个例子中,使用 new 来调用 Person(..) 时,咱们会构造一个新对象(jser)并把它绑定到 Person(..) 调用中的 this 上。
或者能够这么想,谁调用了 getName ?就是 jser 调用了,因此 this 指向了 jser
用 call,apply 和 bind 来修改 this 指向
call 和 apply 是以不一样对象做为上下文对象来调用某个函数,举个例子:
var bar = { name: 'bar', getName () { console.log(this.name) } } var foo = { name: 'foo' } bar.getName.call(foo) // foo
看起来像是借用函数,对象 foo 借用了 bar 的函数 getName,因此咱们判断一个对象类型,常常这么搞:
let foo = [1,2,3,4,5] Object.prototype.toString.call(foo) // "[object Array]"
apply 和 call 的用法同样,不一样点在于 call 用参数表给调用函数传参,而 apply 使用了数组
bind 能够永久性的修改函数中的 this 的指向,不管谁调用,this 指向都同样,并返回了完成绑定的函数,看例子:
var bar = { name: 'bar', getName () { console.log(this.name) } } var foo = { name: 'foo' } foo.func = bar.getName.bind(bar) foo.func() // bar
这里的 func 不受 foo 影响,this 仍是指向了 bar
var bar = { name: 'bar', getName () { console.log(this.name) } } func = bar.getName.bind(bar) func() // bar
这里的 func 也不受 window 影响,this 仍是指向了 bar
综合上述,bind 强制修改了 this,谁调用了函数 this 都不能被修改
若是你把 null 或者 undefined 做为 this 的绑定对象传入 call、apply 或者 bind,这些值再调用时会被忽略,实际应用的是默认绑定。
ES6 中介绍了一种没法使用这些规则的特殊函数类型:箭头函数,根据外层(函数或者全局)做用域来决定 this,箭头函数经常使用于回调函数
ES6 中,let 命令、const 命令、class 命令声明的全局变量,不属于顶层对象的属性,window 没法访问到。var 命令和 function 命令声明的全局变量,属于顶层对象的属性,window 能访问到。
因此 状况 1 中改成:
let name = 'xiaoming' function foo () { console.log(this.name) } foo() // undefined