zoukankan      html  css  js  c++  java
  • 详解 JavaScript的 call() 和 apply()

    详解 JavaScript的 call() 和 apply()

    定义

    ECMAScript规范为所有函数都包含两个方法(这两个方法非继承而来), call 和 apply 。这两个函数都是在特定的作用域中调用函数,能改变函数的作用域,实际上是改变函数体内 this 的值

    call 和 apply
    语法定义说明
    call(thisObj,Object) 调用一个对象的一个方法,以另一个对象替换当前对象 call 方法可以用来代替另一个对象调用一个方法。call 方法可将一个函数的对象上下文从初始的上下文改变为由 thisObj 指定的新对象.如果没有提供 thisObj 参数,那么 Global 对象被用作 thisObj
    apply(thisObj,[argArray]) 应用某一对象的一个方法,用另一个对象替换当前对象 如果 argArray 不是一个有效的数组或者不是 arguments 对象,那么将导致一个 TypeError。如果没有提供 argArray 和 thisObj 任何一个参数,那么 Global 对象将被用作 thisObj, 并且无法被传递任何参数

    用法

    调用函数,传递参数
        //定义一个add 方法
        function add(x, y) {
            return x + y;
        }
    
        //用call 来调用 add 方法
        function myAddCall(x, y) {
            //调用 add 方法 的 call 方法 类似c#中的委托
            return add.call(this, x, y);
        }
    
        //apply 来调用 add 方法
        function myAddApply(x, y) {
            //调用 add 方法 的 applly 方法 类似c#中的委托
            return add.apply(this, [x, y]);
        }
    
        console.log(myAddCall(10, 20));    //输出结果30
      
        console.log(myAddApply(20, 20));  //输出结果40

    我们看到通过方法本身的call 和 apply 执行了该函数。

    改变函数作用域
        var name = '小白';
    
        var obj = {name:'小红'};
    
        function sayName() {
            return this.name;
        }
    
        console.log(sayName.call(this));    //输出小白
    
        console.log(sayName. call(obj));    //输入小红

    我们改变了函数运行的作用域, 通过绑定不同的对象,函数内部 this 也不同。最终输入结果才会这样。

    高级用法,实现 js 继承

        //父类 Person
        function Person() {
            this.sayName = function() {
                return this.name;
            }
        }
    
        //子类 Chinese
        function Chinese(name) {
            //借助 call 实现继承
            Person.call(this);
            this.name = name;
    
            this.ch = function() {
                alert('我是中国人');
            }
        }
    
        //子类 America
        function America(name) {
            //借助 call 实现继承
            Person.call(this);
            this.name = name;
    
            this.am = function() {
                alert('我是美国人');
            }
        }
    
    
        //测试
        var chinese = new Chinese('成龙');
        //调用 父类方法
        console.log(chinese.sayName());   //输出 成龙
    
        var america = new America('America');
        //调用 父类方法
        console.log(america.sayName());   //输出 America

    区别

    • 参数不同, apply 传入的是一个参数数组,也就是将多个参数组合成一个参数数组, call 从第二个参数开始依次传入.
    • apply 可以直接将当前函数的arguments对象作为apply的第二个参数传入

    结束语

    call 和 apply 最大的好处:方便我们解耦,对象不需要和方法有任何的耦合性,能使我们写出更好的面相对象程序。
    大家如果看一些 js 框架底层的话会看到好多地方都有大量用到。

     
     

    其实是一个很简单的东西,认真看十分钟就从一脸懵B 到完全 理解!

    先看明白下面:

      例1

      obj.objAge;  //17

      obj.myFun()  //小张年龄undefined

      例2
     

     shows()  //盲僧 

    比较一下这两者this 的差别,第一个打印里面的this 指向obj,第二个全局声明的shows()函数   this 是window ;

    1,call()、apply()、bind() 都是用来重定义 this 这个对象的!

      如:

      

      obj.myFun.call(db);    //德玛年龄99

        obj.myFun.apply(db);    //德玛年龄99

        obj.myFun.bind(db)();   //德玛年龄99

      以上出了bind 方法后面多了个 () 外 ,结果返回都一致!
      由此得出结论,bind 返回的是一个新的函数,你必须调用它才会被执行
     
      2,对比call 、bind 、 apply 传参情况下
      
      

      obj.myFun.call(db,'成都','上海');     //德玛 年龄 99  来自 成都去往上海

      obj.myFun.apply(db,['成都','上海']);        //德玛 年龄 99  来自 成都去往上海  

      obj.myFun.bind(db,'成都','上海')();         //德玛 年龄 99  来自 成都去往上海

        obj.myFun.bind(db,['成都','上海'])();   //德玛 年龄 99  来自 成都,上海去往undefined

      

       微妙的差距!
      从上面四个结果不难看出
        call 、bind 、 apply 这三个函数的第一个参数都是 this 的指向对象,第二个参数差别就来了:
        call的参数是直接放进去的,第二第三第n个参数全都用逗号分隔,直接放到后面  obj.myFun.call(db,'成都', ... ,'string' );
        apply的所有参数都必须放在一个数组里面传进去  obj.myFun.apply(db,['成都', ..., 'string' ]);
        bind除了返回是函数以外,它 的参数和call 一样。
        
        当然,三者的参数不限定是string类型,允许是各种类型,包括函数 、 object 等等!
         
    posted @ 2017-03-16 17:52 小莫_yu 阅读(21217) 评论(9) 编辑 收藏

  • 相关阅读:
    System.Windows.Forms.Timer与System.Timers.Timer的区别(zz)
    30个最常用css选择器解析(zz)
    Highcharts选项配置详细说明文档(zz)
    CSS For Bar Graphs(maybe old)
    学习CSS3BUTTON(二)
    学习CSS3BUTTON(一)
    CSS CURSOR属性
    CSS3的文字阴影—text-shadow
    display:inline-block; 到底是个啥玩意?
    mysql 子查询
  • 原文地址:https://www.cnblogs.com/qqhfeng/p/10790286.html
Copyright © 2011-2022 走看看