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

var obj1 = {
  username: "HuiTaiLang",
  fn: function() {
    concole.loh(this);
    console.log(this.username);
  }
}
var obj2 = obj1.fn;
obj2(); //undefined

我们想打印对象 obj2 里面的 username 却打印出来 undefined 是怎么回事呢?如果我们直接执行 obj2.fn()是可以的。

var obj1 = {
  username: "HuiTaiLang",
  fn: function() {
    console.log(this);
    console.log(this.username);
  }
}
obj1.fn();   //HuiTaiLang

能够打印是因为,第一个打印里面的 this 指向全局window,第二个打印里面的 this 指向 obj1 。

虽然这种方法可以达到打印username目的,但是如果我们需要将对象保存到另外的一个变量中,那么就可以通过以下方法。

call

var obj1 = {  
  username: "HuiTaiLang",
  fn: function() {
    console.log(this)    
    console.log(this.username);  
  }
}
var obj2 = obj1.fn;
obj2.call(obj1); // HuiTaiLang

通过在 call 方法,第一个参数就是要添加的 obj1环境中,简单来说,this就会指向那个对象。

call 方法除了第一个参数以外还可以添加多个参数,如下:

var obj1 = {  
  username: "HuiTaiLang",
  fn: function(num1, num2) {
    console.log(num1 + num2); 
    console.log(this.username);  
  }
}
var obj2 = obj1.fn;
obj2.call(obj1, 2, 3); // 5

apply

apply方法和call方法有些相似,它也可以改变this的指向。

var obj1 = {  
  username: "HuiTaiLang",
  fn: function() {
    console.log(this.username);  
  }
}
var obj2 = obj1.fn;
obj2.apply(obj1);   // HuiTaiLang

同样apply也可以有多个参数,但是不同的是,第二个参数必须是一个数组,如下:

var obj1 = {  
  username: "HuiTaiLang",
  fn: function(num1, num2) {
    console.log(num1 + num2); 
    console.log(this.username);  
  }
}
var obj2 = obj1.fn;
obj2.apply(obj1, [2, 3]);   // 5

注意:如果call和apply的第一个参数写的是null,那么this指向的是window对象。

var obj1 = {  
  username: "HuiTaiLang",
  fn: function() {
    console.log(this);  
  }
}
var obj2 = obj1.fn;
var obj3 = obj1.fn;
obj2.call(null); // window
obj3.apply(null); // window

bind

bind和call、apply方法不同,实际上bind方法返回的是一个修改过后的函数。

var obj1 = {  
  username: "HuiTaiLang",
  fn: function() {
    console.log(this.username);  
  }
}
var obj2 = obj1.fn;
obj2.bind(obj1);

var newFn = obj2.bind(obj1);
console.log(newFn);
newFn();

我们会发现 bind 并没有被打印,而我们打印 newFn 打印出的是一个函数,并且可以正常执行。

同样bind也可以有多个参数,并且参数可以执行的时候再次添加,但是要注意的是,参数是按照形参的顺序进行的。

var obj1 = {  
  username: "HuiTaiLang",
  fn: function(num1, num2) {
    console.log(num1 + num2);
    console.log(this.username);  
  }
}
var obj2 = obj1.fn;
var newFn1 = obj2.bind(obj1, 2, 3);
var newFn2 = obj2.bind(obj1);
newFn1();
newFn1(2, 3);

总结:

call 、bind 、apply 都可以用来改变上下文中的 this。这三个函数的第一个参数都是 this 的指向对象,后面的参数:

  • call 的参数是直接放进去的,后面的参数全都用逗号分隔。
  • apply 的所有参数,都必须放在一个数组里面传进去。
  • bind 除了返回是函数以外,它的参数和 call 一样。

当然,三者的参数不限定是 string 类型,允许是各种类型,包括函数 、 object 等等!

call、apply 和 bind的更多相关文章

  1. JS核心系列:浅谈 call apply 与 bind

    在JavaScript 中,call.apply 和 bind 是 Function 对象自带的三个方法,这三个方法的主要作用是改变函数中的 this 指向,从而可以达到`接花移木`的效果.本文将对这 ...

  2. Javascript中call,apply,bind方法的详解与总结

    在 javascript之 this 关键字详解 文章中,谈及了如下内容,做一个简单的回顾: 1.this对象的涵义就是指向当前对象中的属性和方法. 2.this指向的可变性.当在全局作用域时,thi ...

  3. JS中call、apply、bind使用指南,带部分原理。

    为什么需要这些?主要是因为this,来看看this干的好事. box.onclick = function(){ function fn(){ alert(this); } fn();}; 我们原本以 ...

  4. Javascript中call、apply、bind函数

    javascript在函数创建的时候除了自己定义的参数外还会自动新增this和arguments两个参数 javascript中函数也是对象,call.apply.bind函数就是函数中的三个函数,这 ...

  5. 图解call、apply、bind的异同及各种实战应用演示

    一.图解call.apply.bind的异同 JavaScript中函数可以通过3种方法改变自己的this指向,它们是call.apply.bind.它们3个非常相似,但是也有区别.下面表格可以很直观 ...

  6. js里function的apply vs. bind vs. call

    js里除了直接调用obj.func()之外,还提供了另外3种调用方式:apply.bind.call,都在function的原型里.这3种方法的异同在stackoverflow的这个答案里说的最清楚, ...

  7. JS之apply,call,bind区别

    为了加深对基础知识的理解,今天再复习下js中的apply,call,bind的区别和用法.整理笔记的过程也是一个再次学习的过程. apply和call js中的调用apply和call方法可以改变某个 ...

  8. JavaScript中call、apply、bind、slice的使用

    1.参考资料 http://www.cnblogs.com/coco1s/p/4833199.html   2.归结如下 apply . call .bind 三者都是用来改变函数的this对象的指向 ...

  9. JS中的控制函数调用:call(),apply()和bind()

    所有的函数都具有call(),apply()和bind()方法.它们可以在执行方法的时候用一个值指向this,并改变面向对象的作用域. apply方法: 以下的两种表达式是等价的: func(arg1 ...

  10. JS中的call、apply、bind方法

    JS中的call.apply.bind方法 一.call()和apply()方法 1.方法定义 call方法: 语法:call([thisObj[,arg1[, arg2[,   [,.argN]]] ...

随机推荐

  1. Properties打印流

    简介 java.util.Properties 继承于 Hashtable ,来表示一个持久的属性集.它使用键值结构存储数据,每个键及其对应值都是一个字符串.该类也被许多Java类使用,比如获取系统属 ...

  2. bom-页面加载事件

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  3. Java线程--ForkJoinPool使用

    原创:转载需注明原创地址 https://www.cnblogs.com/fanerwei222/p/11871099.html Java线程--ForkJoinPool使用 简单解释下: Fork是 ...

  4. iOS WKWebView 使用笔记

    首先添加Webkit框架

  5. 扩展NSDate类实现快捷使用 —— 昉

    获取当前日期和时间: +(NSDate *)getCurrentDate{ NSDate *now = [NSDate date]; return now; } 将日期转换为字符串: +(NSStri ...

  6. Zabbix企业级开源监控解决方案

    Zabbix企业级开源监控解决方案 目录 Zabbix企业级开源监控解决方案 一.Zabbix 1. 监控系统的必要性 2. 监控软件的作用 3. Zabbix的定义 4. Zabbix的监控原理 5 ...

  7. kubernetes基础——1.基本概念

    一.kubernetes特性 自动装箱,自我修复,水平扩展,服务发现和负载均衡,自动发布和回滚,密钥和配置管理,存储编排,批量处理执行. 二.kubernetes cluster Masters * ...

  8. Azure AD Domain Service(二)为域服务中的机器配置 Azure File Share 磁盘共享

    一,引言 Azure File Share 是支持两种认证方式的! 1)Active Directory 2)Storage account key 记得上次分析的 "Azure File ...

  9. webhook触发jenkins进行sonar检测

    目的 jenkins仅需创建一个job,git推送后自动进行sonar代码检测并上传到sonarqube jenkins插件 已按社区推荐安装基本插件 Generic Webhook Trigger ...

  10. Solution -「CF 840C」On the Bench

    \(\mathcal{Description}\)   Link.   给定数列 \(\{a_n\}\),求排列 \(\{p_n\}\) 的个数,使得 \((\forall i\in[1,n))(a_ ...