zoukankan      html  css  js  c++  java
  • JS中的call、apply、bind

    JS中的call、apply、bind都是用来改变函数中this指向的

    box.onclick = function(){
        var name = 1;
      function fn(){
        alert(this.name);
      }
      fn();
          console.log(this.name);
    };

    alert出来的this.name 会显示undefined

    console.log显示出来的this.name为1

    问题来了,函数里边的函数的this不是外层函数的this,而是window。

    一般我们解决这个问题会用

    var _this = this;

    来保存外层函数的this在内层函数中使用。

    但本文将介绍一种更加常用的方法改变this指向。

    box.onclick = function(){
      function fn(){
        console.log(this);
      }
      fn.call(this);
    };

    很神奇吧,call的作用就是改变this的指向的,第一个传的是一个对象,就是你要借用的那个对象。

    fn.call(this); 
      这里的意思是让this去调用fn这个函数,这里的this是box,这个没有意见吧?如果这个你不清楚,很可能你是javscript的新朋友。box调用fn,这句话非常重要,我们知道this它始终指向一个对象,刚好box就是一个对象。那么fn里面的this就是box。

    box.onclick = function(){
      function fn(){
        console.log(this);
      }
      fn.call(this);
    };

    这句话在某些情况下是可以简写的,比如:

    box.onclick = function(){
      var fn = function(){
        console.log(this); //box
      }.call(this);
    };

    或者这样:

    box.onclick = function(){
      (function(){
        console.log(this);
      }.call(this)); //box
    };

    又或者这样:

    var objName = {name:'JS2016'};
    var obj = {
      name:'0 _ 0',
      sayHello:function(){
        console.log(this.name);
      }.bind(objName)
    };
    obj.sayHello();//JS2016

    call和apply、bind都是用来改变this的指向的,但也有一些小小的差别。下面我们来看看它们的差别在哪:

    function fn(a,b,c,d){
      console.log(a,b,c,d);
    }
    //call
    fn.call(null,1,2,3);
    //apply
    fn.apply(null,[1,2,3]);
    //bind
    var f = fn.bind(null,1,2,3);
    f(4);
    
    结果如下:
    1 2 3 undefined
    1 2 3 undefined
    1 2 3 4

      前面说过第一个参数传的是一个你要借用的对象,但这么我们不需要,所有就传了一个null,当然你也可以传其他的,反正在这里没有用到,除了第一个参数后面的参数将作为实际参数传入到函数中。

      call就是挨个传值,apply传一个数组,bind也是挨个传值,但和call和apply还有一些不同,使用call和apply会直接执行这个函数,而bind并不直接执行,而是将绑定好的this重新返回一个新函数,什么时候调用由你自己决定。

    var objName = {name:'JS2016'};
    var obj = {
      name:'0 _ 0',
      sayHello:function(){
        console.log(this.name);
      }.bind(objName)
    };
    obj.sayHello();//JS2016

    这里也就是为什么我要用bind的原因,如果用call的话就会报错了。自己想想这个sayHello在obj都已经执行完了,就根本没有sayHello这个函数了。

    这几个方法使用的好的话可以帮你解决不少问题比如: 
    正常情况下Math.max只能这样用

    Math.max(10,6)

    但如果你想传一个数组的话你可以用apply

    var arr = [1,2,30,4,5];
    console.log(Math.max.apply(null,arr));

      又或者你想让伪数组调用数组的方法

    function fn(){
      [].push.call(arguments,3);
      console.log(arguments); //[1, 2, 3]
    }
    fn(1,2);

    再者:

    var arr = ['aaabc'];
    console.log(''.indexOf.call(arr,'b')); //3

    实际上浏览器内部根本就不在乎你是谁,它只关心你传给我的是不是我能够运行的,如下: 

    正常情况

    var str = 'aaabc';
    console.log(str.indexOf('b'));

    而这种情况其实做的事情和上面模一样,看我来拆解。

    var arr = ['aaabc'];
    ''.indexOf.call(arr);

    这句话就是说让arr调用字符串的indexOf方法,前面说过了浏览器内部不在乎你是谁,所以谁都可以来调用,但不是100%成功,具体看如下。

    ''.indexOf.call(arr,'b')

    这里的arr就是[‘aaabc’],内部很可能拆成了’aaabc’,因此就成了下面的这段代码。

    'aaabc'.indexOf('b');
  • 相关阅读:
    【Python爬虫】第五课(b站弹幕)
    【Python爬虫】第四课(查询照片拍摄地址)
    一些tips
    【Python爬虫】第三课(提取数据)
    【Python爬虫】第二课(请求头设置)
    【Python爬虫】第一课
    【数据分析】如何进行数据分析
    【数据分析】派单排序策略优化验证(附sql查询代码)
    python基础01
    消息
  • 原文地址:https://www.cnblogs.com/guanghe/p/8668467.html
Copyright © 2011-2022 走看看