zoukankan      html  css  js  c++  java
  • 关于setInterval和setTImeout中的this指向问题

    出处:http://www.cnblogs.com/zsqos/p/6188835.html

    前些天在练习写一个小例子的时候用到了定时器,发现在setInterval和setTimeout中传入函数时,函数中的this会指向window对象,如下例:

    复制代码
    var num = 0;
    function Obj (){
        this.num = 1,
        this.getNum = function(){
            console.log(this.num);
        },
        this.getNumLater = function(){
            setTimeout(function(){
                console.log(this.num);
            }, 1000)
        }
    }
    var obj = new Obj; 
    obj.getNum();//1  打印的为obj.num,值为1
    obj.getNumLater()//0  打印的为window.num,值为0
    复制代码

    从上述例子中可以看到setTimeout中函数内的this是指向了window对象,这是由于setTimeout()调用的代码运行在与所在函数完全分离的执行环境上. 这会导致这些代码中包含的 this 关键字会指向 window (或全局)对象。详细可参考MDN setTimeout

    但是在setTimeout中传入的不是函数时,this则指向当前对象,如下例:

    复制代码
    var num = 0;
    function Obj (){
        this.num = 1,
        this.getNum = function(){
            console.log(this.num);
        },
        this.getNumLater = function(){
            setTimeout(console.log(this.num), 1000)
        }
    }
    var obj = new Obj; 
    obj.getNum();//1  打印的为obj.num,值为1
    obj.getNumLater()//1  打印的为obj.num,值为1
    复制代码

    从以上两个例子可以看出,当在setTimeout中传入的参数为函数时,函数内部的this才会指向window对象。

    当在setTimeout中传入了一个函数,若想要让this指向正确的值,可以使用以下两种比较常用的方法来使this指向正确的值:

    1.将当前对象的this存为一个变量,定时器内的函数利用闭包来访问这个变量,如下:

    复制代码
    var num = 0;
    function Obj (){
        var that = this;    //将this存为一个变量,此时的this指向obj
        this.num = 1,
        this.getNum = function(){
            console.log(this.num);
        },
        this.getNumLater = function(){
            setTimeout(function(){
                console.log(that.num);    //利用闭包访问that,that是一个指向obj的指针
            }, 1000)
        }
    }
    var obj = new Obj; 
    obj.getNum();//1  打印的为obj.num,值为1
    obj.getNumLater()//1  打印的为obj.num,值为1
    复制代码

    这种方法是将当前对象的引用放在一个变量里,定时器内部的函数来访问到这个变量,自然就可以得到当前的对象。

    2.利用bind()方法

    复制代码
    var num = 0;
    function Obj (){
        this.num = 1,
        this.getNum = function(){
            console.log(this.num);
        },
        this.getNumLater = function(){
            setTimeout(function(){
                console.log(this.num);
            }.bind(this), 1000)    //利用bind()将this绑定到这个函数上
        }
    }
    var obj = new Obj; 
    obj.getNum();//1  打印的为obj.num,值为1
    obj.getNumLater()//1  打印的为obj.num,值为1
    复制代码

    bind()方法是在Function.prototype上的一个方法,当被绑定函数执行时,bind方法会创建一个新函数,并将第一个参数作为新函数运行时的this。在这个例子中,在调用setTimeout中的函数时,bind方法创建了一个新的函数,并将this传进新的函数,执行的结果也就是正确的了。关于bind方法可参考 MDN bind

    以上两种方法都是比较常用的,当然如果使用call或apply方法来代替bind方法,得到的结果也是正确的,但是call方法会在调用之后立即执行,那样也就没有了延时的效果,定时器也就没有用了,所以推荐使用上述两种方法来将this传进setTimeout和setInterval中。 

  • 相关阅读:
    python实践分享:关于排序算法,怎么选择sort()或者sorted()?
    python面试题汇总第06期-正则表达式(内附7题及答案)
    2020年最全python面试题汇总第05期(内附字符串8题及答案)
    2020年最全python面试题汇总第04期(内附13题及答案)
    2020年最全python面试题汇总第03期(内附10题及答案)
    2020年最全python面试题汇总第02期(内附18题及答案)
    2020年最全python面试题汇总第02期(内附16题及答案)
    python爬虫从小白到高手 Day2 动态页面的爬取
    python爬虫从小白到高手 Day1 爬取百度音乐歌单
    python爬虫从小白到高手 Day0 综述
  • 原文地址:https://www.cnblogs.com/thelongmarch/p/7199672.html
Copyright © 2011-2022 走看看