资讯专栏INFORMATION COLUMN

Javascript中call的使用

Snailclimb / 1351人阅读

摘要:中的使用自己感觉蛮纠结的,根据文档很好理解,其实很难确定你是否真正的理解。将被用作当前对象的对象。方法可将一个函数的对象上下文从初始的上下文改变为由指定的新对象。最基本的理解自定义一个类,该类有一个方法,用来显示当前对象的值。

Javascript中call的使用自己感觉蛮纠结的,根据文档很好理解,其实很难确定你是否真正的理解。 call 方法 应用于:Function 对象 调用一个对象的一个方法,以另一个对象替换当前对象。 call([thisObj[,arg1[, arg2[, [,.argN]]]]]) 参数:thisObj 可选项。将被用作当前对象的对象。 arg1, arg2, , argN 可选项。将被传递方法参数序列。 说明: call 方法可以用来代替另一个对象调用一个方法。call 方法可将一个函数的对象上下文从初始的上下文改变为由 thisObj 指定的新对象。如果没有提供 thisObj 参数,那么 Global 对象被用作 thisObj。

最基本的理解
自定义一个类,该类有一个方法showTxt,用来显示当前对象的name值。创建一个对象,并且该对象的name值等于test1。使用call方法,使新创建的对象obj添加Class1类的showTxt方法,即把Class1类中的this.showTxt中的this指定成obj,这样obj就有了showTxt方法。弹出"test1"。这个例子很容易理解。

</>复制代码

  1. function Class1(){
  2. this.showTxt = function(){alert(this.name)}
  3. }
  4. var obj = new Object();
  5. obj.name="test1"
  6. Class1.call(obj);
  7. obj.showTxt();//test1
  8. alert(obj.showTxt);//function(){alert(this.name)}

再看一个稍微深入的理解
创建Class1的实例,让这个实例调用showTxt方法返回这个实例的name值,因为这个实现没有name值所以返回undefine。

</>复制代码

  1. function Class1(){
  2. this.showTxt = function(){alert(this.name)}
  3. }
  4. var class1 = new Class1();
  5. class1.showTxt();//undefined
  6. alert(class1.showTxt);//function(){alert(this.name)}

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

</>复制代码

  1. function Class1(){
  2. this.name = "class1";//添加name值
  3. this.showTxt = function(){alert(this.name)}
  4. }
  5. var class1 = new Class1();
  6. class1.showTxt();//class1
  7. alert(class1.showTxt);//function(){alert(this.name)}

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

</>复制代码

  1. function Class1(){
  2. this.name = "class1";//添加name值
  3. this.showTxt = function(){alert(this.name)}
  4. }
  5. function Class2(){
  6. this.name = "class2";
  7. }
  8. var class2 = new Class2();
  9. Class1.call(class2);
  10. alert(class2.showTxt);//function(){alert(this.name)}
  11. class2.showTxt();//class1

如果在Class1.call(obj);之后再添加obj.name = "test1",最后结果会输入test1,原因显而易见。

</>复制代码

  1. function Class1(){
  2. this.name = "class1";//添加name值
  3. this.showTxt = function(){alert(this.name)}
  4. }
  5. function Class2(){
  6. this.name = "class2";
  7. }
  8. var class2 = new Class2();
  9. Class1.call(class2);
  10. class2.name = "test1";//重定义obj.name值
  11. alert(class2.showTxt);//function(){alert(this.name)}
  12. class2.showTxt();//test1

上面的例子call的都是一个对象的实例,接下来的案例把对象的实例直接换成函数,看看执行结果会发生哪些变化

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

</>复制代码

  1. function Class1(){
  2. this.showTxt = function(){alert(this.name)}
  3. }
  4. function Class2(){
  5. this.name = "class2";
  6. }
  7. Class1.call(Class2);
  8. alert(Class2.showTxt);//function(){alert(this.name)}
  9. Class2.showTxt();//undefined

接着看下面的例子
class1.showTxt.call(class2);之所以会返回class2是因为function(){alert(this.name)}这里的thiscall指定成了class2,变成了alert(class2.name),所以会返回class2
alert(class2.showTxt)返回undefined,说明并未定义class2.showTxt方法。

</>复制代码

  1. function Class1(){
  2. this.showTxt = function(){alert(this.name)}
  3. }
  4. function Class2(){
  5. this.name = "class2";
  6. }
  7. var class1 = new Class1();
  8. var class2 = new Class2();
  9. class1.showTxt.call(class2);//class2
  10. alert(class2.showTxt);//undefined

因为并为给class2添加showTxt方法,所以提示该错误。如果在这个调用之前添加Class1.call(class2);这个调用就OK了Class1.call(class2);class2.showTxt();//class1

这个例子都会返回undefined

</>复制代码

  1. function Class1(){
  2. this.showTxt = function(){alert(this.name)}
  3. }
  4. function Class2(){
  5. this.name = "class2";
  6. }
  7. var class1 = new Class1();
  8. class1.showTxt.call(Class2);//undefined
  9. alert(Class2.showTxt);//undefined

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

</>复制代码

  1. function add(a,b){
  2. alert(a+b);
  3. }
  4. function sub(a,b){
  5. alert(a-b);
  6. }
  7. add.call(sub,3,1);//4

结果返回4,add.call(sub,3,1)在执行过程中,sub做为add函数中this的替代品出现,但是因为add里没有用到this,所以sub函数直接忽略,所以结果是4。
所以实际执行如下:返回4。

</>复制代码

  1. function add(a,b){
  2. alert(a+b);
  3. }
  4. add(3,1);//4

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

</>复制代码

  1. function f1(){
  2. alert(1);
  3. }
  4. function f2(){
  5. alert(2)
  6. }
  7. var f3 = f1.call;
  8. f1.call(f2);//1
  9. 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的方法:

</>复制代码

  1. function jsCall(oThis){//这里的jsCall就是Call
  2. var argsNew = [];
  3. for(var i=1;i
  4. 或简写成

  5. </>复制代码

    1. function jsCall(oThis){//这里的jsCall就是Call
    2. var argsNew = [].slice.call(arguments,1)
    3. return this.apply(oThis,argsNew);
    4. }
    5. Function.prototype.jsCall = jsCall;
  6. 这样就得到了一个和call一样功能的jsCall, 接下来构建两个函数f1,f2

  7. </>复制代码

    1. function f1(a){
    2. alert([this,a,"f1"]);
    3. }
    4. f1(11);//[object Window],11,f1
    5. function f2(a){
    6. alert([this,a,"f2"]);
    7. }
    8. f2(22);//[object Window],22,f2
  8. jsCallf1中的this替换成f2

  9. </>复制代码

    1. function f1(a){
    2. alert([this,a,"f1"]);
    3. }
    4. function f2(a){
    5. alert([this,a,"f2"]);
    6. }
    7. f1.jsCall(f2,11);//function f2(a){alert([this, a, "f2"]);},11,f1
  10. 执行结果发现[object Window]被替换成f2函数

  11. </>复制代码

    1. function jsCall(oThis){//这里的jsCall就是Call
    2. var argsNew = [].slice.call(arguments,1)
    3. return this.apply(oThis,argsNew);
    4. }
    5. Function.prototype.jsCall = jsCall;
    6. function f1(a){
    7. alert([this,a,"f1"]);
    8. }
    9. function f2(a){
    10. alert([this,a,"f2"]);
    11. }
    12. f1.jsCall.jsCall(f2,11);//11,,f2
  12. 在执行f1.jsCall.jsCall(f2,11);时返回11,,f2,为什么会返回这个结果,重点来了:)
    f1.jsCall方法:

  13. </>复制代码

    1. alert(f1.jsCall);
    2. //返回
    3. //function jsCall(oThis) {
    4. // var argsNew = [].slice.call(arguments, 1);
    5. // return this.apply(oThis, argsNew);
    6. //}
  14. 所以f1.jsCall.jsCall可以替换成jsCall.jsCall看一下执行结果

  15. </>复制代码

    1. function jsCall(oThis){//这里的jsCall就是Call
    2. var argsNew = [].slice.call(arguments,1)
    3. return this.apply(oThis,argsNew);
    4. }
    5. Function.prototype.jsCall = jsCall;
    6. function f1(a){
    7. alert([this,a,"f1"]);
    8. }
    9. function f2(a){
    10. alert([this,a,"f2"]);
    11. }
    12. jsCall.jsCall(f2,11);//11,,f2
  16. 接着分析
    jsCall在执行的过程中,return this.apply(oThis,argsNew);里的this被替换成了
    f211做为参数传给了(oThis,argsNew),变成了f2.apply(11);

  17. </>复制代码

    1. function jsCall(oThis){//这里的jsCall就是Call
    2. var argsNew = [].slice.call(arguments,1)
    3. return this.apply(oThis,argsNew);
    4. }
    5. Function.prototype.jsCall = jsCall;
    6. function f1(a){
    7. alert([this,a,"f1"]);
    8. }
    9. function f2(a){
    10. alert([this,a,"f2"]);
    11. }
    12. f2.apply(11);//11,,f2
  18. 返回结果跟f1.jsCall.jsCall(f2,11)一样。
    回过头来看

  19. </>复制代码

    1. function f1(){
    2. alert(1);
    3. }
    4. function f2(){
    5. alert(2)
    6. }
    7. var f3 = f1.call;
    8. f1.call(f2);//1
    9. f3.call(f2);//2
  20. 这样就不难理解f1.call.call(f2)实现时,f1.call执行过程中call中的thisf2替换成了
    f2.call();因为f2里没有this的引用所以执行结果是2.
    f2.call()//2
    需要十分注意的是f1.call是方法,f1是函数对象,这两者在call时是有区别的。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/78549.html

相关文章

  • Javascriptcall和apply理解

    摘要:剩下的两个,我们分别改变了他们的执行环境,分别指向了和,于是结果就是显示对象各自的值。如果你仍然对和没有清晰的认识,可以试着这样理解。 在Javascript中,每个函数都包含两个非继承而来的方法,call和apply。这两个方法的用途都是在特定的作用域中调用函数,实际上等于设置函数体内的this对象的值。摘自《JavaScript高级程序设计》 apply方法接收两个参数,第一个参数...

    mrcode 评论0 收藏0
  • 理解 JavaScript call()/apply()/bind()

    摘要:理解文章中已经比较全面的分析了在中的指向问题,用一句话来总结就是的指向一定是在执行时决定的,指向被调用函数的对象。与和直接执行原函数不同的是,返回的是一个新函数。这个新函数包裹了原函数,并且绑定了的指向为传入的。 理解 JavaScript this 文章中已经比较全面的分析了 this 在 JavaScript 中的指向问题,用一句话来总结就是:this 的指向一定是在执行时决定的,...

    duan199226 评论0 收藏0
  • 深入浅出JavaScriptcall()、apply()方法

    摘要:的作用在中,方法和方法都是为了改变函数运行时上下文而存在的,换句话说就是为了改变函数体内部的指向。欢迎前端大牛纠正错误,如有错误我会及时改正。 写在前面: 隔了很长时间了,也不知道写点什么。最近一直在研究ES6,一直想写出来的文章能对初学者或者是在学习JS路上有所帮助的。这就是我的初衷。 call、apply的作用 在JavaScript中,call()方法和apply()方法都是为了...

    Cympros 评论0 收藏0
  • 魔幻语言 JavaScript 系列之 call、bind 以及上下文

    摘要:那么,它到底是如何工作的呢让我们从一种更简单的实现开始实际上这种实现代码更短,并且更易读是函数原型中的一个函数,它调用函数,使用第一个参数作为参数,并传递剩余参数作为被调用函数的参数。 原文:The Most Clever Line of JavaScript 作者:Seva Zaikov 原文 最近 一个朋友 发给我一段非常有趣的 JavaScript 代码,是他在某个 开源库中...

    cuieney 评论0 收藏0
  • JavaScript对象理解

    摘要:面向对象编程语言的对象体系,不是基于类的,而是基于构造函数和原型链。一语言使用构造函数作为对象的模板。构造函数之所以叫构造函数,就是说这个函数的目的,就是操作一个空对象即对象,将其构造为需要的样子。 Javascript 前言:关于javascript我总有些疑问,是不是一门面向对象的语言,总结了很久:Javascript是一种基于对象(object-based)的语言,你遇到的所有东...

    aboutU 评论0 收藏0
  • JavaScriptcall()理解

    摘要:返回值使用调用者提供的值和参数调用该函数的返回值。这个匿名函数的主要目的是给每个数组元素对象添加一个方法,这个方法可以打印出各元素在数组中的正确索引号。严格模式下,的值将会是。 Function.prototype.call() 概念 call()方法使用一个指定的this值和单独给出的一个或多个参数来调用一个函数。 注意:该方法的语法和作用与apply()方法类似,只有一个区别,就...

    W_BinaryTree 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<