Hello,今天和大家分享如何修改setInterval作用域。

0.引子

  最近在做一个项目的时候需要开发一个图片轮播显示的组件,在实现过程中遇到了关于setInterval作用域的问题。

  

 SlidePicture.prototype={
constructor:SlidePicture,
init:function(){ //初始化函数
this.autoPlay();
},
slidePic:function(){ //滑动图片具体实现
//省略实现代码
console.log('duration-->'+this.duration);
console.log('delay-->'+this.delay);
},
autoPlay:function(){ //自动轮播图片
setInterval(this.slidePic,this.delay);
}
}
//构造函数
function SlidePicture(delay,duration){
this.delay=delay;
this.duration=duration;
this.init();
} var sp=new SlidePicture(5000,800);

  在上面的代码中,我们自定义并扩展一个SlidePicture原型。主要总用就是,我们在创建原型对象时会调用window的setInterval方法,

  来定时调用图片切换方法,从而实现自动图片的自动轮播效果。

  slidePic方法中具体的实现方法我先省略了(下次可以继续和大家分享这个完整的组件),只是在这里打印了对象中的两个属性值,

  结果却出现以下信息:

  

  都是undefined,为什么呢?

  很明显问题不是出在SlidePicture对象上,因为我们已经给这个对象初始化过属性值了。

  打开firebug脚本调试,在第8行打上断点:

  

  终于发现了问题所在:slidePic方法中的this居然是指向window而不是当前的SlidePicture对象。

  

1.问题

  我们重新审查代码,结果把关注点提到第12行来:

setInterval(this.slidePic,this.delay);

  问题来了,setInterval以及setTimeout这些方法都是全局作用域window的函数,

  所以上面的代码等价于:

 setInterval(function(){
console.log('duration-->'+this.duration);
console.log('delay-->'+this.delay);
},this.delay);

  那么问题就清楚了,函数实现中的this当然是指向window了,千万别被this.slidePic蒙骗。

  既然this就是window,那么delay,duration这两个属性当然也就不存在了。

2.apply/call?

  既然是作用域的问题,那么解决问题的关键也就是如何修改setInterval的作用域了。

   修改函数作用域,我们的第一个方案当然就是apply/call了,我们按照理论尝试将代码修改如下:

autoPlay:function(){    //自动轮播图片
var self=this;
setInterval(self.slidePic.apply(self),this.delay);
}

  我们将当前对象this传递给apply方法,看起来应该是OK的,运行后结果如下:

  

  果然,结果出来了。

  等等,好像有点不对!这个结果只打印了一次,并不是根据setInterval定时循环调用的。

  换成了call结果也一样。

  查了查这两个方法:apply/call如果用在setInterval中的话,只会立即运行一次,而不会按照我们的设想循环调用。

  怎么办?

  我们的主角出场了:bind。

  call和apply是立即执行方法,而bind是产生一个新方法用于后续调用。

  我们修改了下代码:

setInterval(self.slidePic.bind(self),this.delay);

  果然我们想要的结果出现了:

  

  

3.匿名函数

  还有没有别的办法?

  其实在一开始的时候我们已经说了,给setInterval传递this.slidePic参数等价于:

1 setInterval(function(){
2 console.log('duration-->'+this.duration);
3 console.log('delay-->'+this.delay);
4 },this.delay);

  这里我们定义了一个匿名函数作为参数setInterval的第一个参数,但是忽略了一点,

  这个匿名函数中的this仍然指向window,所以这种做法当然也是不行。

  那么我们只要将this换成我们的SlidePicture对象就可以了。

  调整下代码:

autoPlay:function(){    //自动轮播图片
var self=this;
// setInterval(self.slidePic.bind(self),this.delay);
setInterval(function(){
self.slidePic();
},this.delay);
}

  在调用setInterval之前我们先将当前对象this保存下来,然后在匿名函数中调用了就OK了!

  这也是处理this对象的一种常见方案。

   ------写得乱七八糟的,不管怎样希望对各位有所帮助-----

  

修改setInterval作用域的更多相关文章

  1. javascript 欺骗词法作用域

    如果词法作用域完全由写代码期间函数所声明的位置来定义,怎样才能在运行时来"修改"(也可以说欺骗)词法作用域呢?    JavaScript 中有两种机制来实现这个目的.社区普遍认为 ...

  2. JavaScript夯实基础系列(一):词法作用域

      作用域是一组规则,规定了引擎如何通过标识符名称来查询一个变量.作用域模型有两种:词法作用域和动态作用域.词法作用域是在编写时就已经确定的:通过阅读包含变量定义的数行源码就能知道变量的作用域.Jav ...

  3. javasrcipt的作用域和闭包(二)

    这篇博客主要对词法作用域与欺骗词法作用域.函数作用域与块级作用域.函数内部的变量提成原理进行详细的分析,在这篇博客之前,关于作用域.编译原理.浏览器引擎的原理及关系在javaScript的作用域和闭包 ...

  4. AngularJs之Scope作用域

    前言: 上篇博文AngularJs之directive中说了Scope作用域是个大坑,所以拿出来作为重点总结! 什么是scope AngularJS 中,作用域是一个指向应用模型的对象,它是表达式的执 ...

  5. JavaScript:修改作用域外变量

    今天在看JavaScript学习指南的时候做的课后习题,也因此详细的对函数的传入参数进行比较深入的研究. 题目如下: 函数如何才能修改其作用域之外的变量?编写一个函数,由1~5的数字组成的数组作为参数 ...

  6. JS中setInterval、setTimeout不能传递带参数的函数的解决方案

    在JS中无论是setTimeout还是setInterval,在使用函数名作为调用句柄时都不能带参数,而在许多场合必须要带参数,接下来为大家介绍具体的解决方法 在JS中无论是setTimeout还是s ...

  7. 《你不知道的JavaScript》第一部分:作用域和闭包

    第1章 作用域是什么 抛出问题:程序中的变量存储在哪里?程序需要时,如何找到它们? 设计 作用域 的目的:为了更好地存储和访问变量. 作用域:根据名称查找变量的一套规则,用于确定在何处以及如何查找变量 ...

  8. 理解AngularJS的作用域Scope

    AngularJS中,子作用域一般都会通过JavaScript原型继承机制继承其父作用域的属性和方法.但有一个例外:在directive中使用scope: { ... },这种方式创建的作用域是一个独 ...

  9. phpcms后台部分修改

    1.后台登陆前提示信息取消及成功后提示信息取消.    (1)后台登陆前提示信息取消               phpcms\modules\admin\classes\admin.class.ph ...

随机推荐

  1. 破解php-screw加密过的文件有效方法

    今天终于搞定更改过密钥的php-screw解密问题,乐呵一下! 改进下 这样就可以解密任何加密过的PHP源码(包括更改过密钥的),解密的原理稍后具体列出,先说下如何加密 列出之前写使用php scre ...

  2. Html form 表单提交前验证

    可以使用form表单的onsubmit方法,在提交表单之前,对表单或者网页中的数据进行检验. onsubmit指定的方法返回true,则提交数据:返回false不提交数据. 直接看下面的代码: 1 & ...

  3. TCP中 recv和sendf函数

    recv和send函数: #include<sys/socket.h> ssize_t recv(int sockfd, void *buff, size_t nbytes, int fl ...

  4. iOS tableView 选中某个cell时 标准的处理方法

    以前选中cell时,常常判断选中的行数,但是当cell的顺序发生变化时,就要改动处理函数,特别是行数比较多的时候,很麻烦. 之后运用cell的title的内容判断,但是这种判断与现实的内容密切相关,如 ...

  5. php 字符串负值判断

    2014年9月9日 11:54:54 $a = '-1'; $b = (int)$a; $c = is_numeric($a); if ($a) { echo 1; //echo 1 } else { ...

  6. 通过eclipse配置Spring MVC项目

    上一篇刚建立了一个简单的Spring项目,其实Spring MVC是一个和Struts2一样的基于MVC设计模式的web框架,并且继承了MVC的优点,是基于请求驱动的轻量级的web框架,spring ...

  7. 现在, Delphi 的多线程已经非常易用了!

    先看一个非多线程的例子, 代码执行时不能进行其它操作(譬如拖动窗体): {自定义方法: 在窗体上绘制...} procedure MyMethod; var   i: Integer; begin   ...

  8. 浅谈c标签

    今天用c便签,没有写好,以为jsp不兼容c标签,最后发现,原来是c标签写错了,好吧,简单写两句. jsp要用c标签,首先要导入c标签库,导入jstl.jar与standard.jar两个jar包,一般 ...

  9. DOS 循环读取txt每一行内容

    在命令行窗口中输入: for /f %i in (f:\mydata.txt) do echo %i 如果要是写成批处理文件run.bat for /f %%i in (f:\mydata.txt) ...

  10. git merge和个git rebase的区别

    http://stackoverflow.com/questions/16666089/whats-the-difference-between-git-merge-and-git-rebase/16 ...