zoukankan      html  css  js  c++  java
  • js 中的this指向问题

    JS中this上下文对象使用方式

    转载  2016-10-09 

    这篇文章主要为大家详细介绍了JS中this上下文对象使用方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

    JavaScript 有一套完全不同于其它语言的对 this 的处理机制。 在五种不同的情况下 ,this 指向的各不相同。

    有句话说得很在理 -- 谁调用它,this就指向谁

    一、全局范围内

    在全局范围内使用this ,它将指向全局对象(浏览器中为 window)

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    var name = 'name1';
    console.log(name);
     
    this.name = 'name2';
    console.log(name);
    console.log(this.name);
     
    window.name = 'name3';
    console.log(name);
    console.log(this.name);
    console.log(window.name);

    二、函数调用

    直接调用一个函数,this 默认会指向全局 (浏览器端为window)

    1
    2
    3
    4
    5
    6
    7
    8
    var name = 'name1';
    function sayName(){
     console.log(name);
     console.log(this);
    }
     
    sayName();
    window.sayName();

    可以看到

    还有几个常见的情况,根据谁调用方法就指向谁的原则,this的指向要细看

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    // 全局 name
    var name = 'name1';
     
    var obj = {
     name: 'name2',
     sayName: function(){
        // 调用它的时候 this指向全局
      return function(){
       console.log(this.name);
      };
     },
     changeName: function(){
        // 调用它的时候 this指向全局
      setTimeout(function(){
       this.name = 'name3';
      },0);
     }
    };
     
    obj.sayName()();
    obj.changeName();
    setTimeout(function(){
     console.log(name);
     console.log(obj.name);
    },0);

    像这些类似匿名的函数,默认都是被全局(浏览器下的window)对象调用,要正确地让obj调用,就要指代好

    可以用that保持this再进行下一步,或者匿名函数传值,或者使用call/apply/bind改变context等

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    var name = 'name1';
     
    var obj = {
     name: 'name2',
     sayName: function(){
      var that = this;
      return function(){
       console.log(that.name);
      };
     },
     changeName: function(){
      var that = this;
      setTimeout(function(){
       that.name = 'name3';
      },0);
     }
    };
     
    obj.sayName()(); // name2
    obj.changeName();
    setTimeout(function(){
     console.log(name); // name1
     console.log(obj.name); // name3
    },0);

    三、作为对象方法的调用

    其实就类似上头提到的 obj.sayName()  obj.name 等

    这时this会指向这个obj 

    四、call/apply/bind 的调用

    当使用 Function.prototype 上的 call 或者 apply ,bind 方法时,函数内的 this将会被 显式设置为函数调用的第一个参数。

    具体使用方法

    我们可以稍微修改一下上头的代码,就可以看到this指向的改变

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    var name = 'name1';
     
    var obj = {
     name: 'name2',
     sayName: function(){
      // 返回一个默认全局的函数
      return function(){
       console.log(this.name);
      };
     },
     changeName: function(){
      // 返回一个默认全局的函数
      setTimeout(function(){
       this.name = 'name3';
      // 然后将该函数绑定给this(当前obj对象)
      }.bind(this),0);
     }
    };
     
    // obj.sayName()这个函数,让obj来调用
    obj.sayName().call(obj);
    // 让this(也就是全局对象)来调用
    obj.sayName().apply(this);
     
    obj.changeName();
    setTimeout(function(){
     // 输出更改之后,全局name的值
     console.log(name);
     // 输出更改之后,obj对象中 name的值
     console.log(obj.name);
    },0);

    五、作为构造函数调用

    比如 new Foo();

    先来看个简单的例子:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    var name = 'name1';
    function Foo(){
     // 赋值this(当前对象)的name属性值
     this.name = 'name2';
    }
     
    // new 构造函数产生一个实例
    var foo = new Foo();
     
    console.log(name);
    console.log(foo.name);
     
    // 直接调用该函数
    Foo();
    console.log(name);

    可以看到,如果函数倾向于和 new 关键词一块使用,则我们称这个函数为构造函数,当new 了之后,this则指向这个心创建的对象(这个new 的过程其实也涉及到了继承机制)。

    若直接调用这个函数,this就默认执行全局对象了。

    JS中this上下文对象使用方式

    转载  2016-10-09   作者:imwtr    我要评论

    这篇文章主要为大家详细介绍了JS中this上下文对象使用方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

    JavaScript 有一套完全不同于其它语言的对 this 的处理机制。 在五种不同的情况下 ,this 指向的各不相同。

    有句话说得很在理 -- 谁调用它,this就指向谁

    一、全局范围内

    在全局范围内使用this ,它将指向全局对象(浏览器中为 window)

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    var name = 'name1';
    console.log(name);
     
    this.name = 'name2';
    console.log(name);
    console.log(this.name);
     
    window.name = 'name3';
    console.log(name);
    console.log(this.name);
    console.log(window.name);

    二、函数调用

    直接调用一个函数,this 默认会指向全局 (浏览器端为window)

    1
    2
    3
    4
    5
    6
    7
    8
    var name = 'name1';
    function sayName(){
     console.log(name);
     console.log(this);
    }
     
    sayName();
    window.sayName();

    可以看到

    还有几个常见的情况,根据谁调用方法就指向谁的原则,this的指向要细看

     
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    // 全局 name
    var name = 'name1';
     
    var obj = {
     name: 'name2',
     sayName: function(){
        // 调用它的时候 this指向全局
      return function(){
       console.log(this.name);
      };
     },
     changeName: function(){
        // 调用它的时候 this指向全局
      setTimeout(function(){
       this.name = 'name3';
      },0);
     }
    };
     
    obj.sayName()();
    obj.changeName();
    setTimeout(function(){
     console.log(name);
     console.log(obj.name);
    },0);

    像这些类似匿名的函数,默认都是被全局(浏览器下的window)对象调用,要正确地让obj调用,就要指代好

    可以用that保持this再进行下一步,或者匿名函数传值,或者使用call/apply/bind改变context等

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    var name = 'name1';
     
    var obj = {
     name: 'name2',
     sayName: function(){
      var that = this;
      return function(){
       console.log(that.name);
      };
     },
     changeName: function(){
      var that = this;
      setTimeout(function(){
       that.name = 'name3';
      },0);
     }
    };
     
    obj.sayName()(); // name2
    obj.changeName();
    setTimeout(function(){
     console.log(name); // name1
     console.log(obj.name); // name3
    },0);

    三、作为对象方法的调用

    其实就类似上头提到的 obj.sayName()  obj.name 等

    这时this会指向这个obj 

    四、call/apply/bind 的调用

    当使用 Function.prototype 上的 call 或者 apply ,bind 方法时,函数内的 this将会被 显式设置为函数调用的第一个参数。

    具体使用方法

    我们可以稍微修改一下上头的代码,就可以看到this指向的改变

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    var name = 'name1';
     
    var obj = {
     name: 'name2',
     sayName: function(){
      // 返回一个默认全局的函数
      return function(){
       console.log(this.name);
      };
     },
     changeName: function(){
      // 返回一个默认全局的函数
      setTimeout(function(){
       this.name = 'name3';
      // 然后将该函数绑定给this(当前obj对象)
      }.bind(this),0);
     }
    };
     
    // obj.sayName()这个函数,让obj来调用
    obj.sayName().call(obj);
    // 让this(也就是全局对象)来调用
    obj.sayName().apply(this);
     
    obj.changeName();
    setTimeout(function(){
     // 输出更改之后,全局name的值
     console.log(name);
     // 输出更改之后,obj对象中 name的值
     console.log(obj.name);
    },0);

    五、作为构造函数调用

    比如 new Foo();

    先来看个简单的例子:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    var name = 'name1';
    function Foo(){
     // 赋值this(当前对象)的name属性值
     this.name = 'name2';
    }
     
    // new 构造函数产生一个实例
    var foo = new Foo();
     
    console.log(name);
    console.log(foo.name);
     
    // 直接调用该函数
    Foo();
    console.log(name);

    可以看到,如果函数倾向于和 new 关键词一块使用,则我们称这个函数为构造函数,当new 了之后,this则指向这个心创建的对象(这个new 的过程其实也涉及到了继承机制)。

    若直接调用这个函数,this就默认执行全局对象了。

  • 相关阅读:
    beeframework开发笔记1
    CentOS 6.0最小化编译安装Nginx+MySQL+PHP+Zend
    (转)Android-Mac电脑如何进行APK反编译-使用apktool、jd-gui
    (转)【Android测试工具】03. ApkTool在Mac上的安装和使用(2.0版本)
    淘宝PHPSDK2.0 剔除 lotusphp框架---兄弟连教程
    (转载)postgresql navicat 客户端连接验证失败解决方法:password authentication failed for user
    (转载)CentOS6下 源代码方式安装openERP7.0
    在阿里云 centos 6.3上面安装php5.2(转)
    php自动转换pfx到pem和cer(dem格式)到pem
    WebSocket获取httpSession空指针异常的解决办法
  • 原文地址:https://www.cnblogs.com/pengwenfan/p/8833094.html
Copyright © 2011-2022 走看看