一,三者共同点

js中的apply,call,bind是对于初学者比较难的概念之一,比如说我。。参考几篇文章之后,统一来讲,

1、这三个函数都属于Function.prototype下面的方法,如下图所示。从而可以被每一个函数实例所调用。

2、他们的作用都是改变函数的执行上下文。举个例子:

var a=1;
function printA(){
console.log(this.a);
}
var obj={a:2};
printA();//相当于window.printA(),此时this指向window/global对象

现在我想打印obj里面的a,可以这么做:

obj.printA=printA;//将函数作为obj对象的方法
obj.printA();//调用该方法

还可以这么做:

printA.call(obj); //利用call将obj作为printA的上下文,this指向obj
printA.apply(obj); //利用apply将obj作为printA的上下文,this指向obj
var objPrintA=printA.bind(obj); //利用bind将obj作为printA的上下文,this指向obj,并让objPrintA变量名作为返回的新函数的引用
objPrintA();

上面的三种方法都成功的访问到了obj里的a;

但是注意bind的用法,printA.bind(obj)仅仅将printA的上下文改变,并返回一个新函数,称为绑定函数。但是并没有立刻执行,需要手动调用新函数objPrintA来执行它。

3、call、apply、bind都可以在第一个参数后传入别的参数,作为函数的参数传入

第一个参数除了普通的对象之外,还可以传入以下值:

(1) 不传,或者传入null,undefined,函数中的this指向window对象。

(2)传递另一个函数的函数名,函数的this指向这个函数的引用。

(3)传递字符串、数值或者布尔值等基础类型,函数中的this指向其对应的包装对象,String,Number,Boolean

(4)传递一个对象,函数中的this指向该对象。

function printThis(){
console.log(this);
}
printThis.call(); //[object Window]
printThis.call(null); //[object Window]
printThis.call(undefined);//[object Window]
printThis.call(1);//[object Number]
printThis.call('1');//[object String]
printThis.call(true);//[object Boolean]
printThis.call({a:1});//[object Object]{a: 1}
printThis.call(function(a){console.log(a)});//function (a){console.log(a)}

二,call与apply的区别

前面区分了bind与call,apply的区别,bind会返回一个新的绑定上下文之后的函数,而call和apply是会立即执行函数。下面的例子实现了es3对原生bind()方法的模拟,更能说明这个区别。

function bind(f,o){
if(f.bind) return f.bind(o);//如果有原生bind,则直接调用
else return function(){//返回一个新的函数,调用这个函数会在o对象的上下文中执行
return f.apply(o,arguments);
}
}

call和apply的区别在于apply接受一个数组参数,所以当你的参数是明确知道数量时用 call,当参数不固定时应该用apply 。一个应用情景就是arguments和apply比较配

arguments是函数的一个内部属性:

function a(){
console.log(arguments);
console.log(a.arguments);
console.log(this.arguments);
}
obj={arguments:'arguments'}
a();
/*
[object Arguments]{length: 0}
[object Arguments]{length: 0}
undefined
*/
a(1,'1',true);
/*
[object Arguments]{0: 1, 1: "1", 2: true, length: 3}
[object Arguments]{0: 1, 1: "1", 2: true, length: 3}
undefined
*/
a.call(obj);
/*
[object Arguments]{length: 0}
[object Arguments]{length: 0}
arguments
*/

上面这个例子表明arguments是函数对象的一个属性,在函数内部可以直接通过arguments[i]来访问实参。还说明了this绝对不是指向函数本身。

最后举一个求平方和的例子。

function sum(){
var args=Array.prototype.slice.apply(arguments);
return args.reduce(function(prev,next){return prev+next*next},0);
}
sum(1,2,4,6,8)//121

参考文章:深入浅出 妙用Javascript中apply、call、bind

理解javascript里的ABC--apply bind call的更多相关文章

  1. JavaScript学习(2)call&apply&bind&eval用法

    javascript学习(2)call&apply&bind&eval用法 在javascript中存在这样几种特别有用的函数,能方便我们实现各种奇技淫巧.其中,call.bi ...

  2. Learning JavaScript with MDN (call, apply, bind)

    Learning JavaScript with MDN (call, apply, bind) call, apply, bind Object.prototype.toString() 检测 js ...

  3. 深入浅出:了解JavaScript中的call,apply,bind的差别

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

  4. 别真以为JavaScript中func.call/apply/bind是万能的!

    自从学会call/apply/bind这三个方法后我就各种场合各种使用各种得心应手至今还没踩过什么坑,怎么用?说直白点就是我自己的对象没有某个方法但别人有,我就可以通过call/apply/bind去 ...

  5. javascript中的call(),apply(),bind()方法的区别

    之前一直迷惑,记不住call(),apply(),bind()的区别.不知道如何使用,一直处于懵懂的状态.直到有一天面试被问到了这三个方法的区别,所以觉得很有必要总结一下. 如果有不全面的地方,后续再 ...

  6. 深入理解javascript中的Function.prototye.bind

    函数绑定(Function binding)很有可能是你在开始使用JavaScript时最少关注的一点,但是当你意识到你需要一个解决方案来解决如何在另一个函数中保持this上下文的时候,你真正需要的其 ...

  7. 理解javascript中的Function.prototype.bind

    在初学Javascript时,我们也许不需要担心函数绑定的问题,但是当我们需要在另一个函数中保持上下文对象this时,就会遇到相应的问题了,我见过很多人处理这种问题都是先将this赋值给一个变量(比如 ...

  8. 理解 JavaScript 中的 Function.prototype.bind

    函数绑定(Function binding)很有可能是你在开始使用JavaScript时最少关注的一点,但是当你意识到你需要一个解决方案来解决如何在另一个函数中保持this上下文的时候,你真正需要的其 ...

  9. 理解JavaScript里this关键字

    1.全局代码中的this:始终指向window 2.函数代码中的this: }; var bar = { x: , test: function () { alert(this === bar); a ...

随机推荐

  1. 灾难 bzoj 2815

    灾难(1s 128MB)catas [样例输入] 5 0 1 0 1 0 2 3 0 2 0 [样例输出] 4 1 0 0 0 题解: 主要算法:拓扑排序:最近公共祖先(Lca): 先跑出拓扑序 我们 ...

  2. DSP的Gel作用

    转自:http://blog.csdn.net/azhgul/article/details/6660960 最近刚在研究Davinci系,特此MARK下,以资后续学习之用. DSP的Gel作用 1 ...

  3. 浅析java内存模型--JMM(Java Memory Model)

    在并发编程中,多个线程之间采取什么机制进行通信(信息交换),什么机制进行数据的同步? 在Java语言中,采用的是共享内存模型来实现多线程之间的信息交换和数据同步的. 线程之间通过共享程序公共的状态,通 ...

  4. Nginx配置文件nginx.conf中文详解(转)

    ######Nginx配置文件nginx.conf中文详解##### #定义Nginx运行的用户和用户组 user www www; #nginx进程数,建议设置为等于CPU总核心数. worker_ ...

  5. ERDAS 2014安装破解及汉化图文教程

    关于ERDAS 2014 的安装破解及其汉化过程: 说明:汉化只是之针对英语能力不能正常满足使用软件的朋友们.如果英语过专八,完全没有必要汉化!在此之前查了好多关于ERDAS汉化的文档,可惜只是简单的 ...

  6. 【web前端面试题整理08】说说最近几次面试(水)

    为什么换工作 换工作简单来讲一般会归纳为钱不够或者人不对,我们团队氛围很不错,所以基本就定位到钱不够了,而我更多是考虑到以后的职业发展,简单说来就是对以后几年的工作有想法,而这种想法实现不一定能在现在 ...

  7. CentOS7 查看IP、Gateway、DNS、Hostname

    1.查看IP# ip addr 2.查看路由# ip route 3.查看DNS# cat /etc/resolv.conf 4.查看主机名# hostname

  8. DAO设计模式

    DAO设计模式 DAO设计模式简介: DAO设计模式可以减少代码量,增强程序的可移植性,提高代码的可读性. DAO(数据库操作对象)设计模式是 JavaEE 数据层的操作.主要由五部分组成: 1.数据 ...

  9. Java 性能分析工具 , 第 2 部分:Java 内置监控工具

    引言 本文为 Java 性能分析工具系列文章第二篇,第一篇:操作系统工具.在本文中将介绍如何使用 Java 内置监控工具更加深入的了解 Java 应用程序和 JVM 本身.在 JDK 中有许多内置的工 ...

  10. git与github安装、配置、pull、push

    操作系统是Ubuntu 16.04 LTS 64bit 1 安装git (1)安装 sudo apt-get install git-core (2)一些全局变量的初始化 在本地建立一个文件夹,然后做 ...