来源:http://ibruce.info/2013/11/22/hexo-your-blog/
call 方法
应用于:Function 对象
调用一个对象的一个方法,以另一个对象替换当前对象。
1、最基本的理解:
示例1
自定义一个类,该类有一个方法showTxt,用来显示当前对象的name值。
创建一个对象,并且该对象的name值等于test1。
使用call方法,使新创建的对象obj添加Class1类的showTxt方法,即把Class1类中的this.showTxt中的this指定成obj,这样obj就有了showTxt方法。弹出”test1”。
这个例子很容易理解。
2、再看一个稍微深入的理解
示例2: 创建Class1的实例,让这个实例调用showTxt方法返回这个实例的name值,因为这个实现没有name值所以返回undefine.
示例3: 下面就给Class1添加个name值,这时class1再调用showTxt方法,会返回class1,这是因为给类添加了name值,所以实例的name也由undefine变成了class1.
示例4: Class1.call(obj) 这个操作把Class1中的this.name,this.showTxt里的this替换成了obj,所以就变成了obj.name=’class1’,所以obj.showTxt在执行时返回class1。
示例5: 如果在Class1.call(obj);之后再添加obj.name = ‘test1’,最后结果会输入test1,原因显而易见。
上面的例子call的都是一个对象的实例,接下来的案例把对象的实例直接换成函数,看看执行结果会发生哪些变化
3、把call方法的第一参数由实例换成函数看看会怎么
示例6: Class2是一个function对象的引用,在执行Class1.call(Class2)时this.showTxt里的this被替换成了Class2。这样Class2就有了showTxt方法,Class2.showTxt()执行时会返回Class2.name的值,因为Class2并未定义name值,所以会返回undefined。
Class2函数里的this.name是由Class2创建实例的name值,并不是Class2.name,这两个值有时会让我迷糊。
4、接着看下面的例子
示例7: class1.showTxt.call(class2);之所以会返回class2是因为function(){alert(this.name)}这里的this被call指定成了class2,变成了alert(class2.name),所以会返回class2.
alert(class2.showTxt)返回undefined,说明并未定义class2.showTxt方法。
因为并为给class2添加showTxt方法,所以提示该错误。如果在这个调用之前添加Class1.call(class2);这个调用就OK了
示例8: 这个例子都会返回undefined
5、在使用call时如果调用函数里没有this会怎么样
示例9:
结果返回4,add.call(sub,3,1)在执行过程中,sub做为add函数中this的替代品出现,但是因为add里没有用到this,所以sub函数直接忽略,所以结果是4。
所以实际执行如下:返回4。
6、不错,接下来再理解一个怪异的形式
示例10:
在理解f1.call.call(f2)时我们首先要知道call方法到底是如何执行的,这样才能f1.call.call(f2)如何执行。
示例11:
引用JK写的一个用apply实现call的方法:
这样就得到了一个和call一样功能的jsCall.
接下来构建两个函数f1,f2
用jsCall把f1中的this替换成f2
执行结果发现[object Window]被替换成f2函数
在执行f1.jsCall.jsCall(f2,11);时返回11,,f2,为什么会返回这个结果,重点来了:)
f1.jsCall方法:
所以f1.jsCall.jsCall可以替换成jsCall.jsCall看一下执行结果
接着分析
jsCall在执行的过程中,return this.apply(oThis,argsNew);里的this被替换成了f2,11做为参数传给了(oThis,argsNew),变成了f2.apply(11);
返回结果跟f1.jsCall.jsCall(f2,11)一样。
回过头来看
这样就不难理解f1.call.call(f2)实现时,f1.call执行过程中call中的this被f2替换成了f2.call();因为f2里没有this的引用所以执行结果是2.
需要十分注意的是f1.call是方法,f1是函数对象,这两者在call时是有区别的。