欢迎各位兄弟 发布技术文章

这里的技术是共享的

You are here

Javascript中call的使用 有大用

Javascript中call的使用自己感觉蛮纠结的,根据文档很好理解,其实很难确定你是否真正的理解。

call 方法
应用于:Function 对象
调用一个对象的一个方法,以另一个对象替换当前对象。
call([thisObj[,arg1[, arg2[,   [,.argN]]]]])
参数:
thisObj 
可选项。将被用作当前对象的对象。 
arg1, arg2, , argN 
可选项。将被传递方法参数序列。 
说明:
call 方法可以用来代替另一个对象调用一个方法。call 方法可将一个函数的对象上下文从初始的上下文改变为由 thisObj 指定的新对象。如果没有提供 thisObj 参数,那么 Global 对象被用作 thisObj。

1.最基本的理解:示例1

自定义一个类,该类有一个方法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)}

这个例子很容易理解。

2.再看一个稍微深入的理解

示例2:创建Class1的实例,让这个实例调用showTxt方法返回这个实例的name值,因为这个实现没有name值所以返回undefine.


function Class1(){
this.showTxt = function(){alert(this.name)}
}

var class1 = new Class1();
class1.showTxt();
//undefined
alert(class1.showTxt);//function(){alert(this.name)}

示例3:下面就给Class1添加个name值,这时class1再调用showTxt方法,会返回class1,这是因为给类添加了name值,所以实例的name也由undefine变成了class1.


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)}

示例4:Class1.call(obj) 这个操作把Class1中的this.name,this.showTxt里的this替换成了obj,所以就变成了obj.name='class1',所以obj.showTxt在执行时返回class1。


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

示例5:如果在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的都是一个对象的实例,接下来的案例把对象的实例直接换成函数,看看执行结果会发生哪些变化

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,这两个值有时会让我迷糊。


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  //应该是类本身的默认的名字 Class2

4.接着看下面的例子

示例7:class1.showTxt.call(class2);之所以会返回class2是因为function(){alert(this.name)}这里的this被call指定成了class2,变成了alert(class2.name),所以会返回class2.

alert(class2.showTxt)返回undefined,说明并未定义class2.showTxt方法。

 


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

示例8:这个例子都会返回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

5.在使用call时如果调用函数里没有this会怎么样

示例9:


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

6.不错,接下来再理解一个怪异的形式

示例10:


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)如何执行。

示例11:

引用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时是有区别的。

来自  http://hszy00232.blog.163.com/blog/static/43022753201131835653841/

 

  登录后你可以发表评论,请先登录。登录>>
2015-07-12 23:55
看到 [].slice.call(arguments,1) 时头都大了!call的用法越来越迷糊了……算了,老婆催我睡觉了,下个周末再来拜读了
2014-10-03 23:03
christ
示例8中 貌似结果不对
function Class1(){
this.showTxt = function(){alert(this.name)}
}
function Class2(){
this.name = 'class2';
}
var class1 = new Class1();
class1.showTxt.call(Class2);//这里应该是 class2
alert(Class2.showTxt);//undefined
2015-07-12 21:52
zhaiduting 回复 christ

测试一下就知道了,的确是 undefined

2015-07-12 13:31

受益匪浅,感谢博主!对了,发现示例7最后一行有处笔误。应该显示 class2 而非 class1

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

Class1.call(class2);
class2.showTxt();    //显示 class2 而非 class1

2013-08-21 10:11
由浅入深,例子很清晰呀。谢谢分享。
 只是那个call.call 很让人费解。实际应用中的价值不大,博主这样写只是助于理解call apply的真正意思吧。
2013-04-18 18:50
新手,虽然写的很详细,但我还是有些地方看不懂,不知道楼允许加QQ加,交流一下吗
 
2012-12-07 18:06
园子里有一篇文章,简易浅明,很好懂,和楼主有异曲同工之妙。结尾一句话必杀技:当参数明确时可用call, 当参数不明确时可用apply给合arguments

http://www.cnblogs.com/fighting_cp/archive/2010/09/20/1831844.html
2013-01-18 16:58
长缨 回复 冰封骑士

2013-01-04 00:20
大经哥
不错啊。!!!
2012-11-23 18:17
sddad
示例6 中Class2.showTxt();返回的怎么是类名Class2不是undefined
2012-08-17 19:39
call的各种用法统一性在于:obj.call(obj2) 用obj2来替代obj函数“里面”的那个this指针。obj是一个函数对象的引用,是一个函数名,是可以生成新对象的函数名。
2012-08-17 19:28

很好,我觉得,函数对象和自定义对象之间的关系确实比较容易混淆,所以我总结了一句话:函数对象(objiect Function)的引用就是自定义对象(defined object)的构造函数的名字。function Class1(){   } Class1是一个Function object的引用 Class1又是一个函数名,同时可以生成新的对象,var obj = new Class1();

obj是Class1对象的一个引用,但是它不能再是一个构造函数的名字,例如function obj(){} 这应该是不允许的。

2012-03-05 20:04
sss
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
 
上面的例子我用firefox测试后发现Class2.showTxt();执行后的值为'class2'
2012-03-05 20:11
sss 回复 sss

看错了,不好意思

2012-08-06 15:55
十八岁的天空 回复 sss
我都测试了,Class2.showTxt()执行的值还是为class2呀。。。。
 
2012-08-09 12:15
tan766 回复 sss
2012-08-09 12:19
tan766 回复 sss

因为Class2并未定义name值,所以会返回undefined。

Class2函数里的this.name是由Class2创建实例的name值,并不是Class2.name,这两个值有时会让我迷糊

晕,高手啊还能鼓捣出一个自圆其说的解释,哈哈
2012-08-17 04:20
916593064 回复 tan766
这并没有很神奇的,Class2是Function对象的一个引用,同时它又是另外一个类的名字,这意味着可以生成Class2对象,你现在领悟到javascript一切皆对象的含义了吧!


来自  http://hszy00232.blog.163.com/blog/static/43022753201131835653841/
普通分类: