博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
轻松搞定javascript中this的指向
阅读量:6399 次
发布时间:2019-06-23

本文共 4969 字,大约阅读时间需要 16 分钟。

  关于javascript中this指向的问题,现总结如下,如有不正确,欢迎指正。

  javascript中,this的指向并不是在函数定义的时候确定的,而是在其被调用的时候确定的。也就是说,函数的调用方式决定了this指向。记住:this 就是一个指针,指向我们调用函数的对象。

  在此将javascript中this的调用方式分为以下几种:

1、直接调用:

  直接调用是指通过 funName(..) 这种方式调用。此时,函数内部的this指向全局变量。

function foo() {    console.log(this === global);}foo(); //true

  注意:直接调用并不是仅仅指在全局作用域下进行调用,而是说在任何作用域下通过funName(..) 这种方式调用。如下两个例子亦是属于直接调用方式:

  a、使用bind函数改变外层函数的作用域,然后在内层直接调用,其this指向依然是全局变量:

function foo() {    console.log(this === global);}function foo1() {    foo();};var foo2 = foo1.bind({}); //改变foo1函数的作用域foo2(); //true 依然指向全局变量

  b、函数表达式,将一个函数赋值给一个变量,然后通过直接调用的调用方式调用此函数,其this指向依然是全局变量:

var obj = {    foo: function(){        console.log(this === obj);   //false        console.log(this === global);//true    }}var f = obj.foo;f();

2、方法调用:

  方法调用是指通过对象来调用其方法函数,类似于obj.foo(..)的调用方式。此时,this指向调用该方法的对象,注意,是最终调用该方法的对象。

var obj = {    foo1: function(){        console.log(this === obj);    }}obj.foo2 = function() {    console.log(this === obj);}function foo3() {    console.log(this === obj);}obj.foo3 = foo3;obj.foo1(); // trueobj.foo2(); // trueobj.foo3(); // true

3、new 调用:

  在es5中,通过 new Constructor() 的形式调用一个构造函数,会创建这个构造函数实例,而这个实例的this指向创建的这个实例。如下例所示,在构造函数内部使用this.name并没有改变全局变量name的值。

var name = "全局";function Person(name){    this.name = name;}var p = new Person("局部");console.log(p.name);console.log(name);

上述三种情况是常见的调用方式,以下还有一些特殊的调用方式:如bind、call、apply以及es6中箭头函数。

4、bind函数对this的影响

  bind函数用于绑定this的指向,并且返回一个绑定函数,此函数绑定了this指向,注意:绑定函数的this指向将不可再次更改。

var obj = {};function foo1() {    console.log(this === obj);}foo1(); //false  此时属于上述直接调用的方式,因此其this指向globalvar foo2 = foo1.bind(obj);foo2(); //true 绑定函数的this指向其绑定的对象/** * 注意:绑定函数的this指向不可更改 */var foo3 = foo2.bind({'a': 1});foo3(); //true foo2.apply({'a': 1}); //true foo2.call({'a': 1});  //true

5、apply和call对this指向的影响

  apply和call亦可以用于改变this指向,并且返回执行结果。关于bind、apply和call的区别以及实现可以参考我的博客:,在此不做重复说明。需要注意的一点是:apply和call不可以改变绑定函数(使用bind返回的函数)的this指向。

var obj = {};function foo1() {    console.log(this === obj);}var foo2 = foo1.bind({'a': 1});/** * 注意:此处并不是绑定函数,因此其返回值依然是ture,即apply改变其this指向。 */foo1.apply(obj); //true/** * 注意:此处是绑定函数,因此不可再通过apply和call的形式改变其this指向。 */foo2.apply(obj); //false

6、es6箭头函数中的this

  箭头函数没有自己的this绑定,其使用的this是其直接父级函数的this。也就是说,箭头函数内部的this是由其直接外层函数(方法)决定的,而外层函数中的this是由其调用方式决定的。

const obj = {    foo: function() {        const inner = () => {            console.log(this === obj);        };        inner();    },    far: function() {        return () => {            console.log(this === obj);        }    }}/** * inner()内的this是foo的this,其指向取决于foo的调用方式 */obj.foo(); //truevar foo1 = obj.foo;foo1();    //false 此时应该指向globalconst far1 = obj.far();far1();    //trueconst far2 = obj.far;far2()();  //false 此时应该指向global

6.1 箭头函数常见错误及其解决方案:

  1. 在对象上定义函数:

    const test = {    array: [1, 2, 3],    sum: () => {        console.log(this === window); // => true        return this.array.reduce((result, item) => result + item);    }};test.sum();// TypeError: Cannot read property 'reduce' of undefined

    原因就是,箭头函数没有它自己的this值,箭头函数内的this值继承自外围作用域。

    对象方法内的this指向调用这个方法的对象,如果使用箭头函数,this和对象方法在调用的时候所处环境的this值一致。因为 test.sum()是在全局环境下进行调用,此时this指向全局。

    解决方法也很简单,使用函数表达式或者方法简写(ES6 中已经支持)来定义方法,这样能确保 this 是在运行时是由包含它的上下文决定的。

    const test = {    array: [1, 2, 3],    sum() {        console.log(this === test); // => true        return this.array.reduce((result, item) => result + item);    }};test.sum();// 6
  2. 定义原型方法

    在对象原型上定义函数也是遵循着一样的规则

    function Person(name) {    this.name = name;}Person.prototype.sayName = () => {    console.log(this === window); // => true    return this.name;};const cat = new Person('Mew');cat.sayName(); // => undefined

    使用传统的函数表达式就能解决问题

    function Person(name) {    this.name = name;}Person.prototype.sayName = function() {    console.log(this === Person); // => true    return this.name;};const cat = new Person('Mew');cat.sayName(); // => Mew
  3. 定义事件回调函数

    thisJS中非常强大的特点,他让函数可以根据其调用方式动态的改变上下文,然后箭头函数直接在声明时就绑定了this对象,所以不再是动态的。

    在客户端,在DOM元素上绑定事件监听函数是非常普遍的行为,在DOM事件被触发时,回调函数中的this指向该DOM,但是,箭头函数在声明的时候就绑定了执行上下文,要动态改变上下文是不可能的,在需要动态上下文的时候它的弊端就凸显出来:

    const button = document.getElementById('myButton');button.addEventListener('click', () => {    console.log(this === window); // => true    this.innerHTML = 'Clicked button';});

    因为这个回调的箭头函数是在全局上下文中被定义的,所以他的thiswindow。换句话说就是,箭头函数预定义的上下文是不能被修改的,这样 this.innerHTML 就等价于 window.innerHTML,而后者是没有任何意义的。

    使用函数表达式就可以在运行时动态的改变 this

    const button = document.getElementById('myButton');button.addEventListener('click', function() {    console.log(this === button); // => true    this.innerHTML = 'Clicked button';});
  4. 定义构造函数

    如果使用箭头函数会报错。

    显然,箭头函数是不能用来做构造函数。

    const Message = (text) => {    this.text = text;};const helloMessage = new Message('Hello World!');// Throws "TypeError: Message is not a constructor"

    理论上来说也是不能这么做的,因为箭头函数在创建时this对象就绑定了,更不会指向对象实例。

箭头函数带来了很多便利。恰当的使用箭头函数可以让我们避免使用早期的.bind()函数或者需要固定上下文的地方并且让代码更加简洁。

箭头函数带来了很多便利。恰当的使用箭头函数可以让我们避免使用早期的.bind()函数或者需要固定上下文的地方并且让代码更加简洁。

  以上是个人对JavaScript中this的理解,欢迎您的留言。

欢迎访问了解更多信息。

参考文献:

1、《》
2、《》

转载地址:http://yhyea.baihongyu.com/

你可能感兴趣的文章
CYQ.Data 快速开发之UI(赋值、取值、绑定)原理
查看>>
使用路由器实现不同VLAN间通信_路由交换
查看>>
真我新格调 勇敢使梦想触手可及
查看>>
ExtJs 动态RadioGroup的子项 [ Ext | RadioGroup | items ]
查看>>
idea模板注释
查看>>
DHCP服务介绍
查看>>
Hibernate的数据删除,更改
查看>>
编程学习初体验(5. 如何自学编程)(2)
查看>>
思科ISR G1与ISR G1C的区别
查看>>
利用perl提取web配置文件中的域名对应的路径
查看>>
VMware虚拟化---集群高可用纵览
查看>>
Centos5上安装JRE和LUMAQQ
查看>>
Ping不通不代表主机不在线!
查看>>
SCCM2007安装部署指南
查看>>
关于监控工具的主动发起性能测试
查看>>
AngularJs $resource 高大上的数据交互
查看>>
Cisco基础实验回顾4--IP classless
查看>>
上接扩展GridView控件(6) - 响应行的单击事件和双击事件
查看>>
负载均衡实施 应该因地制宜
查看>>
非常好的BASH脚本编写教程
查看>>