Javascript中call的使用本身感受蛮纠结的,根据文档很好理解,其实很难肯定你是否真正的理解。 call 方法 应用于:Function 对象 调用一个对象的一个方法,以另外一个对象替换当前对象。 call([thisObj[,arg1[, arg2[, [,.argN]]]]])
参数:thisObj
可选项。将被用做当前对象的对象。 arg1, arg2, , argN
可选项。将被传递方法参数序列。 说明: call 方法能够用来代替另外一个对象调用一个方法。call 方法可将一个函数的对象上下文从初始的上下文改变为由 thisObj 指定的新对象。若是没有提供 thisObj 参数,那么 Global 对象被用做 thisObj。app
最基本的理解
自定义一个类,该类有一个方法showTxt,用来显示当前对象的name值。建立一个对象,而且该对象的name值等于test1。使用call方法,使新建立的对象obj添加Class1类的showTxt方法,即把Class1类中的this.showTxt中的this指定成obj,这样obj就有了showTxt方法。弹出"test1"。这个例子很容易理解。函数
function Class1(){ this.showTxt = function(){alert(this.name)} } var obj = new Object(); obj.name="test1" Class1.call(obj); obj.showTxt();//test1 alert(obj.showTxt);//function(){alert(this.name)}
再看一个稍微深刻的理解
建立Class1的实例,让这个实例调用showTxt方法返回这个实例的name值,由于这个实现没有name值因此返回undefine。this
function Class1(){ this.showTxt = function(){alert(this.name)} } var class1 = new Class1(); class1.showTxt();//undefined alert(class1.showTxt);//function(){alert(this.name)}
下面就给Class1
添加个name
值,这时class1
再调用showTxt
方法,会返回class1
,这是由于给类添加了name
值,因此实例的name
也由undefine
变成了class1
.prototype
function Class1(){ this.name = 'class1';//添加name值 this.showTxt = function(){alert(this.name)} } var class1 = new Class1(); class1.showTxt();//class1 alert(class1.showTxt);//function(){alert(this.name)}
Class1.call(obj)
这个操做把Class1
中的this.name
,this.showTxt
里的this
替换成了obj
,因此就变成了obj.name='class1'
,因此obj.showTxt
在执行时返回class1
。code
function Class1(){ this.name = 'class1';//添加name值 this.showTxt = function(){alert(this.name)} } function Class2(){ this.name = 'class2'; } var class2 = new Class2(); Class1.call(class2); alert(class2.showTxt);//function(){alert(this.name)} class2.showTxt();//class1
若是在Class1.call(obj);
以后再添加obj.name = 'test1'
,最后结果会输入test1
,缘由显而易见。对象
function Class1(){ this.name = 'class1';//添加name值 this.showTxt = function(){alert(this.name)} } function Class2(){ this.name = 'class2'; } var class2 = new Class2(); Class1.call(class2); class2.name = 'test1';//重定义obj.name值 alert(class2.showTxt);//function(){alert(this.name)} class2.showTxt();//test1
上面的例子call
的都是一个对象的实例,接下来的案例把对象的实例直接换成函数,看看执行结果会发生哪些变化ip
把call
方法的第一参数由实例换成函数看看会怎么样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
,这两个值有时会让我迷糊。文档
function Class1(){ this.showTxt = function(){alert(this.name)} } function Class2(){ this.name = 'class2'; } Class1.call(Class2); alert(Class2.showTxt);//function(){alert(this.name)} Class2.showTxt();//undefined
接着看下面的例子class1.showTxt.call(class2);
之因此会返回class2是由于function(){alert(this.name)}
这里的this
被call
指定成了class2
,变成了alert(class2.name)
,因此会返回class2
。 alert(class2.showTxt)
返回undefined
,说明并未定义class2.showTxt
方法。io
function Class1(){ this.showTxt = function(){alert(this.name)} } function Class2(){ this.name = 'class2'; } var class1 = new Class1(); var class2 = new Class2(); class1.showTxt.call(class2);//class2 alert(class2.showTxt);//undefined
由于并为给class2
添加showTxt
方法,因此提示该错误。若是在这个调用以前添加Class1.call(class2);
这个调用就OK了Class1.call(class2);class2.showTxt();//class1
function
这个例子都会返回undefined
function Class1(){ this.showTxt = function(){alert(this.name)} } function Class2(){ this.name = 'class2'; } var class1 = new Class1(); class1.showTxt.call(Class2);//undefined alert(Class2.showTxt);//undefined
在使用call时若是调用函数里没有this
会怎么样
function add(a,b){ alert(a+b); } function sub(a,b){ alert(a-b); } add.call(sub,3,1);//4
结果返回4,add.call(sub,3,1)
在执行过程当中,sub
作为add
函数中this
的替代品出现,可是由于add
里没有用到this
,因此sub
函数直接忽略,因此结果是4。
因此实际执行以下:返回4。
function add(a,b){ alert(a+b); } add(3,1);//4
不错,接下来再理解一个怪异的形式
function f1(){ alert(1); } function f2(){ alert(2) } var f3 = f1.call; f1.call(f2);//1 f3.call(f2);//2
f1.call(f2);
比较好理解,若是不理解看上边的case,但如何理解f3.call(f2)
会返回2呢,为了方便理解进行一下等效变化为f1.call.call(f2)
,这时会发现其实是f1.call
方法call
调用了f2
,那f1
怎么又会有call
方法呢?call
, apply
都属于Function.prototype
的一个方法,它是JavaScript
引擎内在实现的,由于属于Function.prototype
,因此每一个Function
对象实例,也就是每一个方法都有call
, apply
属性。
在理解f1.call.call(f2)
时咱们首先要知道call
方法究竟是如何执行的,这样才能f1.call.call(f2)
如何执行。
引用JK写的一个用apply实现call的方法:
function jsCall(oThis){//这里的jsCall就是Call var argsNew = []; for(var i=1;i<arguments.length;i++){ argsNew.push(arguments[i]); } return this.apply(oThis,argsNew); } Function.prototype.jsCall = jsCall;
或简写成
function jsCall(oThis){//这里的jsCall就是Call var argsNew = [].slice.call(arguments,1) return this.apply(oThis,argsNew); } Function.prototype.jsCall = jsCall;
这样就获得了一个和call
同样功能的jsCall
, 接下来构建两个函数f1
,f2
function f1(a){ alert([this,a,'f1']); } f1(11);//[object Window],11,f1 function f2(a){ alert([this,a,'f2']); } f2(22);//[object Window],22,f2
用jsCall
把f1
中的this
替换成f2
function f1(a){ alert([this,a,'f1']); } function f2(a){ alert([this,a,'f2']); } f1.jsCall(f2,11);//function f2(a){alert([this, a, "f2"]);},11,f1
执行结果发现[object Window]
被替换成f2
函数
function jsCall(oThis){//这里的jsCall就是Call var argsNew = [].slice.call(arguments,1) return this.apply(oThis,argsNew); } Function.prototype.jsCall = jsCall; function f1(a){ alert([this,a,'f1']); } function f2(a){ alert([this,a,'f2']); } f1.jsCall.jsCall(f2,11);//11,,f2
在执行f1.jsCall.jsCall(f2,11);
时返回11,,f2
,为何会返回这个结果,重点来了:)f1.jsCall
方法:
alert(f1.jsCall); //返回 //function jsCall(oThis) { // var argsNew = [].slice.call(arguments, 1); // return this.apply(oThis, argsNew); //}
因此f1.jsCall.jsCall
能够替换成jsCall.jsCall
看一下执行结果
function jsCall(oThis){//这里的jsCall就是Call var argsNew = [].slice.call(arguments,1) return this.apply(oThis,argsNew); } Function.prototype.jsCall = jsCall; function f1(a){ alert([this,a,'f1']); } function f2(a){ alert([this,a,'f2']); } jsCall.jsCall(f2,11);//11,,f2
接着分析 jsCall
在执行的过程当中,return this.apply(oThis,argsNew);
里的this
被替换成了f2
,11
作为参数传给了(oThis,argsNew)
,变成了f2.apply(11);
function jsCall(oThis){//这里的jsCall就是Call var argsNew = [].slice.call(arguments,1) return this.apply(oThis,argsNew); } Function.prototype.jsCall = jsCall; function f1(a){ alert([this,a,'f1']); } function f2(a){ alert([this,a,'f2']); } f2.apply(11);//11,,f2
返回结果跟f1.jsCall.jsCall(f2,11)
同样。
回过头来看
function f1(){ alert(1); } function f2(){ alert(2) } var f3 = f1.call; f1.call(f2);//1 f3.call(f2);//2
这样就不难理解f1.call.call(f2)
实现时,f1.call
执行过程当中call
中的this
被f2
替换成了f2.call();
由于f2
里没有this
的引用因此执行结果是2.f2.call()//2
须要十分注意的是f1.call
是方法,f1
是函数对象,这二者在call
时是有区别的。