栗子最实在:用实例来说明call、apply和bind方法

前言

call,apply和bind方法,之前使用的时候,总会有蒙圈的时候。这次顺着用实例来说明this的含义之后,就一次也把这三个方法整理明白吧。

一、call和apply的比较

ECMAScript标准中Function原型具备有call方法 Function.prototype.call和apply方法Function.prototype.apply,因此JavaScript中每个函数都有call方法和apply方法。
  call方法和apply方法,实现的效果作用是一样的,区别就在于参数的形式和数量上不同。具体而言是:

  • call方法接收的参数数量可以是多个。第一个参数是函数执行上下文的对象,后面的参数可以是多个
<script>
        var a={
            name:'hyh'
        }
        var name='hyhaa'
        function exp(a,b){
            console.log(a+this.name+b)
        }
        exp.call(null,'happy ',' day') //happy hyhaa day
        exp.call(a,'happy ',' day') //happy hyh day
    </script>
  • apply方法接收的参数数量是两个。第一个参数也是函数执行上下文的对象,第二个参数是多个传入的函数参数组成的类数组。
<script>
        var a={
            name:'hyh'
        }
        var name='hyhaa'
        function exp(a,b,c){
            console.log(this.name+a+b+c)
        }
        exp.apply(null,[' happy ',' every',' day']) //hyhaa happy  every day
    </script>

call和apply方法的使用,其实就是换一种参数传递的形式来达到使用函数方法的目的。比如有些时候,我们手头有的参数形式是数组(类数组),直接传递给某个函数方法,但其不接受,那我们就借位,把函数方法换个apply写法,这样就能接受我们的参数形式。

参数为数组的情况下,Math.max方法的使用示例和对比

所以,鉴于apply和call方法的作用是一样的,那么如何挑选哪种写法合适,就看具体情况了。简单来说,就是参数是数组形式,就使用apply方法,参数是单独多个的话,就使用call方法。
  注意:如果第一个参数不需要写,不需要改动函数上下文的对象的话,记得写上null值来占位。如上面例子中的exp.apply(null,[' happy ',' every',' day'])写法

二、call和apply的作用

2.1 改变this的指向

接上面所说的,call方法和apply方法的第一个参数都是作为函数执行上下文的对象,也就是说call方法和apply方法改变了函数被调用时this的指向

    <script>
        var value=100
        var obj5={
            value:200
        }
        function fn4(a,b){
            console.log(this.value+a+b)
        }
        fn4(3,4) //107
        fn4.call(obj5,3,4) //207
        fn4.apply(obj5,[3,4]) //207
    </script>

这个例子中,等同于obj5这个对象传递给了函数fn4,函数fn4的this就指向obj5。等同于实现:

function fn4(a,b){
  console.log(obj5.value+a+b)
}

再举个复杂点的例子:

<script>
    var fruits={
     name:'apple',
     saleprice:function(price){
           console.log(this.name+' price is '+price +' yuan')
         }
     }
     var vegtable={
         name:'tomato'
     }
     fruits.saleprice('20')  //apple price is 20 yuan
     fruits.saleprice.call(vegtable,'12')  //tomato price is 12 yuan
</script>

在这个例子中,fruits.saleprice.call(vegtable,'12')是将vegtable传给到fruits对象中,因此this.name就等同于vegtable的name属性'tomato'。

2.2 立即执行函数

这两个方法的使用,会让函数立即执行。

function exp(){
  console.log(1)
}
exp.call()
exp.apply()

不过因为例如exp()的写法也能让函数立即执行,所以常见的是这种更简便的写法exp()。

三、call和apply常见的应用写法

3.1 数组的扩充

   <script>
        var a=[123,'2fds','3sdf',34]
        var b=['aa','bb',343]
        Array.prototype.push.apply(a,b)
        console.log(a)  //[123, "2fds", "3sdf", 34, "aa", "bb", 343]
   </script>

3.2 字符串的拼接

function expjoin(arguments){
  var a=Array.prototype.join.apply(arguments);
  console.log(a);
}
expjoin([123,'2fds','3sdf',34]) //"123,2fds,3sdf,34"

这个写法也等同于

[].join.apply([123,'2fds','3sdf',34])  //"123,2fds,3sdf,34"

3.3 最大/小值的获取

var arr=[34,54,656,877]  //undefined
Math.max.apply(Math,arr)  //877
Math.max.apply(null,arr)  //877
Math.max.call(null,34,54,656,877)  //877
Math.min.call(null,34,54,656,877)  //34
Math.min.apply(null,arr)  //34
Math.min.apply(Math,arr)  //34

3.4 验证是否为数组

<script>
        function isArray(obj){
            return Object.prototype.toString.call(obj) ==='[object Array]'
        }
        console.log(isArray(343));
        console.log(isArray([4343,565]));
</script>

3.5 把一个类数组转换为真正的数组

var arr= Array.prototype.slice.call([2,4,5]);
arr  //[2, 4, 5]
arr.push('33')  //[2, 4, 5, "33"],能够具备并使用数组的方法例如push()
domNodes.push(3434)  //[2, 4, 5, "33", 3434]

四、call和apply写法的转换

如果看了感觉会晕,那么可以换种方式来看call和apply的写法

Array.prototype.join.apply(arguments)等同于arguments.join()
Array.prototype.slice.call(arguments)等同于arguments.slice()

前面例子中的fruits.saleprice.call(vegtable,'12')等同于vegtable.saleprice('12')

这样就不难理解,诸如下面的写法:

function log(){
    console.log.apply(console, arguments); //console.log(arguments)
}
log(1,2,3,4,5)  //arguments会默认使用[ ]运算符获取参数,结果为:1 2 3 4 5

对上面这个例子使用到了arguments,再来引申讲下arguments

  • 在函数调用时,会自动在该函数内部生成一个名为 arguments的隐藏对象

  • 该对象类似于数组,可以使用[ ]运算符获取函数调用时传递的实参

  • 只有函数被调用时,arguments对象才会创建,未调用时其值为null

 function fn5(name, age){
     console.log(arguments); //["Byron", 20]
     name = 'XXX';
     console.log(arguments); //["XXX", 20]
     arguments[1] = 30;
     console.log(arguments); //["XXX", 30]
 }
 fn5('Byron', 20);

五、bind方法

终于写到bind了~~bind的用法稍微迂回点,需要稍长点耐心理解。
call方法和apply方法是ECMAScript3标准定义的,而bind方法是更晚的ECMAScript5标准定义的方法。

5.1 bind、call和apply方法的相同之处

前面我们已经说了call和apply的作用相同,区别就在参数的形式不同。bind方法的作用也是改变this的指向。bind方法的参数要求和call方法一样,第一个参数是函数执行上下文的对象,后面的参数可以是多个。
  但bind又有自己的特别之处,下面来说说bind的用法上的区别:

5.2 bind方法的用法上的区别

区别一: bind()方法是会创建一个新函数,当调用这个新函数时,会以创建新函数时传入 bind()方法的第一个参数作为 this。
看栗子来体会:

var name='xxx'; 
var obj = {
    name: 'yyy'
}
function func() {
    console.log(this.name);
}
var func1 = func.bind(obj);
func1();   //yyy
func();     //xxx

func1等于通过bind方法创建的和func函数相同的新函数。func1的this对象为传入的obj,所以this.name就只等于obj对象的name值‘yyy’。
  因为是新生成的函数,所以原来函数func()并不受影响,所以直接执行func(),this指向全局window,所以this.name为'xxx'。

区别二: 参数的使用上,call方法是将第二个及之后的参数,作为实参传递到函数中。
  而 bind() 方法的第二个以及以后的参数,再加上新函数运行时的参数,按照顺序作为实参传递到新函数中。
  这样看文字解释,真的太绕了,直接看栗子,就很好理解:

function func(a, b, c) {
    console.log(a, b, c);
}
var func1 = func.bind(null,'aaa');

func('A', 'B', 'C');            // A B C
func1('A', 'B', 'C');           // aaa A B
func1('B', 'C');                // aaa B C
func.call(null, 'aaa');      // aaa undefined undefined

区别三: 最后一个不同,就是bind函数不是立即调用的,实现生成新的处理函数,然后再执行。而call和apply方法都是立即执行。所以写法上,bind函数需要再加个()
  还是看栗子吧,改写下前面的例子:(一样的实现效果,不一样的执行写法)

var name='xxx'; 
var obj = {
    name: 'yyy'
}
function func() {
    console.log(this.name);
}
console.log(func.bind(obj());  //yyy
console.log(func.call(obj));    //yyy
console.log(func.apply(obj));  //yyy
5.3 bind方法常见的应用写法

例子1:改变setTimeout()方法的this指向
  上一篇专门写this的使用实例的时候,讲到一个setTimeout()的例子。上一篇的例子中是使用var _this=this的方式来使setTimeout()的this不指向window。现在学了bind,同样也可以用bind来达到同样效果:

<body>
    <button class="bindtest">dianwo</button>
    <script>
    /*bind的用法--和使用var _this=this达到同样的目的*/
        $bindtest=document.querySelector('.bindtest')
        $bindtest.addEventListener('click',function(){
            console.log(this) //<button class="domtest">dianwo</button>
            setTimeout(function(){
                console.log(this)   //<button class="domtest">dianwo</button>
            }.bind(this),300)
        })
    </script>
</body>

前面说了,bind的作用是得到一个新的函数,而新函数的this就是bind传递的第一个参数。这里bind方法是和setTimeout方法同级的,属于$bindtest对象的点击事件下的,所以this是$bindtest对象。

例子2:字符串拼接的bind方法实现
  前面call方法的使用例子中实现的字符串拼接,现在换成用bind方法来实现:
  常见的是,比如我需要一个join操作,但是我没有这个方法,那么我就需要借助数组Array原型的join方法来借位使用。

function joinStr(){
      var joins=Array.prototype.join.bind(arguments);
      console.log(joins('-'))
}
joinStr('a','b','c')

例子中的写法等同于['a','b','c'].join('-')

六、结语

写了这么一大通,结合实例来理解,真的事半功倍。我列举的call,apply和bind方法的常见使用的例子,只是我目前看到的常见写法。更多的变化写法,就待实践和学习,来扩充积累啦。

参考文章:

https://segmentfault.com/a/1190000009650716
http://book.jirengu.com/fe/%E5%89%8D%E7%AB%AF%E8%BF%9B%E9%98%B6/%E9%9D%A2%E5%90%91%E5%AF%B9%E8%B1%A1/this.html
https://github.com/lin-xin/blog/issues/7
https://segmentfault.com/a/1190000006993545

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容