近来,学习了一下《JavaScript精粹》,读到了函数这章,理清了JavaScript中this在不同调用模式下的指向。
1、Function调用模式:Function是JavaScript的一种引用类型,拥有四种调用模式:方法调用模式,函数调用模式,构造器调用模式,apply(call)调用模式
2、Function的不同调用模式对this产生不同的影响:
(1)方法调用模式:Function作为对象的方法被调用。此时,this指向调用Function的对象。
1 var obj = { 2 name: "lucy", 3 printName: function( ) { 4 console.log( this.name ); // 此时printName,作为obj的方法被调用,this指向调用printName的obj,this.name === "lucy"。 5 } 6 };
(2)构造器调用模式:Function作为构造器被调用,亦即使用new操作符,创建新的对象实例,将新的对象连接到该函数的prototype上,并且将this指向新创建的对象实例;同时,执行构造器内的代码为新的对象实例添加成员,最后返回这个新的对象实例。
1 // 创建构造器Demo,在Demo被new操作符调用时,this指向Demo的prototype。 2 var Demo = function( ){ 3 this.name = "jack"; 4 }; 5 6 // 在Demo的prototype中,添加方法printName。 7 Demo.prototype.printName = function( ){ 8 concole.log(this.name); 9 }; 10 11 // 创建Demo实例,此时this已经指向Demo的prototype 12 var d = new Demo( ); 13 console.log(d.name); // 在控制台输出"jack" 14 d.printName( ); // 在控制台输出"jack"
PS:构造函数、构造器只是在英译中的过程中的翻译用词差异,在英文都是constructor。《JavaScript精粹》译为构造器,《JavaScript高级程序设计》第三版译为构造函数。
(3)apply(call)调用模式:JavaScript中,函数也是对象而具有方法,其具有方法apply。apply接受两个参数,第一个是要传递给this的值,第二个是参数数组。当函数调用apply方法时,将函数内部的this指向apply传递的第一个参数。
1 var people = { 2 this.name = "lily"; 3 } 4 5 var printName = function( ){ 6 console.log(this.name) 7 } 8 9 // 此时apply将this指向了people 10 printName.apply(people) // 在控制台输出 lily
PS:《JavaScript精粹》中只提及apply。apply和call在功能上是相同的,但是具体的使用方式上,有小差异。
(4)函数调用模式:既不是作为方法调用,也不是调用自己的apply(call),也不是作为构造器被new调用,这时候调用就是函数调用模式。此时函数内部的this指向全局变量,在浏览器中指的是window变量,在node.js中指的是global变量。
1 window.name = "bob"; 2 3 // 此时this指向window 4 var printName = function( ) { 5 console.log( this.name ); 6 }; 7 8 console.log(window.name) // 在控制台输出bob 9 printName() // 在控制台输出bob
3、小结:
(1)方法调用模式:this指向调用方法的对象。
(2)构造器调用模式:this指向构造器的prototype。
(3)apply调用模式:this指向apply传入的第一个对象。
(4)函数调用模式:this指向全局对象window。