zoukankan      html  css  js  c++  java
  • [Effective JavaScript 笔记]第25条:使用bind方法提取具有确定接收者的方法

    js里方法和属性值为函数,就像一个东西两种称呼一个样,比如土豆,也叫马铃薯,一个样。既然一样,那就可以对对象的方法提取出来为函数,然后把提取出来的函数作为回调函数直接传递给高阶函数。

    高阶函数是什么

    玩过套娃娃游戏没,没玩过,没事,我也没玩过。
    大致就是下面这个样子
    1464860371895
    呃,好吧,这才是真正的。
    1464860406270
    就是多层函数,以函数为参数或返回值的函数。有点绕,没事看看上面的图就明白了。想了解怎么实现个简单的请点这里
    好了,函数拿出来了,给高阶函数做参数传进去了。
    这里面很容易会忘记把传进去的函数绑定到当前对象上,自由惯了,没办法,自由很重要,但没有绑定的对象,你什么都做不了。比如没有女朋友,你就得靠双手了,伤身呀。

    一个示例

    来看看下面这段代码提提神。

    var you={
        gF:[],
        add:function(s){
            this.gF.push(s);
        },
        all:function(){
            return this.gF.join("-");
        }
    }
    

    比如你不希望一次只交一个女朋友,想同时交往多个。ES5的forEach方法,在每个源数组(多个女朋友)元素上重复地调用add方法(交往),就可以把多个女孩子加到你的后宫了。

    var girls=["西施","王昭君","貂蝉","杨贵妃"];//按人物历史出场顺序,呵呵
    girls.forEach(you.add);
    

    想想看着四大美人,来出来一见。

    you.all();//error:Cannot read property 'push' of undefined

    什么都有人呢,都哪去了。
    因为you.add的接收者并不是你(you对象)。函数的接收者取决于它是如何被调用的,上面并没有调用它,只是把它传给了高阶函数forEach。但forEach的实现使用全局对象,这个时候你加女朋友变成了,在全世界(window对象)里找gF属性,因为这个gF没有为undefined,所以也就没有push方法,这就报错了。
    问题找到了,美女们还往哪里跑~

    var girls=["西施","王昭君","貂蝉","杨贵妃"];//按人物历史出场顺序,呵呵
    girls.forEach(you.add,you);
    you.all();//"西施,王昭君,貂蝉,杨贵妃"

    完美运行,是不是想想都美了。

    匿名函数

    但不是每个高阶函数都像forEach这样善解人意,提供一个其回调函数的接收者。如果遇到个不解风情的函数怎么办?我们可以创建一个调用对象方法的函数来运行。法子如下

    var girls=["西施","王昭君","貂蝉","杨贵妃"];//按人物历史出场顺序,呵呵
    girls.forEach(function(s){
        you.add(s);
    });
    you.all();//"西施,王昭君,貂蝉,杨贵妃"

    好了,事情就这样解决了。

    bind方法

    NO,还有东西要讲,现在请bind方法全场,它是在ES5标准库才有的函数方法。主要作用就是为函数指定一个对象为其接收者,你也可以理解为,就是指定函数里的this指向哪个对象的。现在就看看上面的情况,bind方法如何解决:

    var girls=["西施","王昭君","貂蝉","杨贵妃"];//按人物历史出场顺序,呵呵
    girls.forEach(you.add.bind(you));
    you.all();//"西施,王昭君,貂蝉,杨贵妃"

    怎么样,是不是很牛掰。那bind都弄了啥呢?这里you.add.bind(you)生成了一个全新的函数,这个函数行为和you.add是一致的,但它里把this定死了,死心踏地地就是you了,原来的函数you.add的接收者保持不变。

    you.add===you.add.bind(you);//false

    这样就可以在很多的场合共享函数了。特别是里面有this关键词的原型方法。只要使用bind就再也不用担心this的指向问题了。不用在外层作用域用变量存储this了。

    var obj={
        txt:'hello',
        sayHello:function(){
           console.log(this.txt);
        }
    }
    document.body.onclick=obj.syaHello.bind(obj);//"hello"
    

    事件绑定再也不用担心了。
    bind方法在ES5之前需要兼容写法,详细请点击查看
    下面就又到了书上的提示内容了,记住一小点受用很多噢。

    提示

    • 注意,提取一个方法不会将方法的接收者绑定到该方法的对象上

    • 当给高阶函数传递对象方法时,使用匿名函数在适当的接收者上调用该方法

    • 使用bind方法创建绑定到适当接收者的函数

    附录:这次没有附录,本节里的内容,在之前都有,没有引进新的知识点。

  • 相关阅读:
    Kafka中的数据清理(logdeletion)
    genymotion虚拟器笔记
    layui hover显示图片
    vue 导出到excel
    el-tree知识积累
    js 含有对象的数组去重
    苏宁易购价格爬取(golang)
    vue+elementui+beego笔记
    vue笔记
    beego笔记
  • 原文地址:https://www.cnblogs.com/wengxuesong/p/5555714.html
Copyright © 2011-2022 走看看