一直以来对bind()、apply()、call()这三个方法都模模糊糊的,现在有时间详细的看看这三个方法,并记录下来。

bind()

参考文档:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Function/bind

  • 定义:

bind()方法创建一个新的函数, 当被调用时,将其this关键字设置为提供的值,在调用新函数时,在任何提供之前提供一个给定的参数序列。

  • 用法:

func.bind(thisArg, arg1, arg2, ...)

  1. var obj = {
  2. a: 1,
  3. b: 2,
  4. getCount: function(c, d) {
  5. return this.a + this.b + c + d;
  6. }
  7. };
  8. window.a = window.b = 0;
  9. var func1 = obj.getCount;
  10. var func2 = obj.getCount.bind(obj);
  11. console.log(func1(3, 4));//得到7
  12. console.log(func2(3, 4));//得到10
  • 说明:

bind()是Function的一个方法,所以应该是一个a函数来调用bind,参数是可以有若干个如(d,e,f),调用了bind方法后会返回一个新的方法b,b = a.bind(c,d,e,f),新函数b调用时,实际调用的却是a,但this指针指向的的却是c,d,e,f会作为参数带到a函数里面;
在上例中,func1之所以得到7是因为func1的this指向的是window,func2调用时this指向的是obj,所以得到10。

  • 示例:
  1. 创建绑定函数(如上例):

    1. (function() {
    2. console.log(this.a);
    3. }.bind({
    4. a: 10
    5. }))()//输出10
    6. (function() {
    7. console.log(this.a);
    8. })()//输出undefined
  2. 偏函数:创建一个调用另一个部分——参数或变量已经预置的的函数——的函数的用法(拗口,看示例吧),说白了就是让一个函数拥有预设的初始参数
    1. //例1
    2. function list() {
    3. return Array.prototype.slice.call(arguments);
    4. }
    5. var list1 = list(1, 2, 3); // [1, 2, 3]
    6. var leadingThirtysevenList = list.bind(undefined, 37);//37作为list的预设的初始参数
    7. var list2 = leadingThirtysevenList(); // [37]
    8. var list3 = leadingThirtysevenList(1, 2, 3); // [37, 1, 2, 3]
    9. //例2
    10. function joinWords(a,b) {
    11. return [a,b].join(' ');
    12. }
    13. //不使用bind
    14. function prefixer(word) {
    15. return function(b) {
    16. return joinWords(word,b);
    17. }
    18. }
    19. var prefixWithHate = prefixer('Hate');
    20. console.log(prefixWithHate('Java'));//Hate Java
    21. //使用bind
    22. var prefixWithHate2 = joinWords.bind(window,"Hate");
    23. console.log(prefixWithHate2('Java'));//Hate Java
  3. 配合 setTimeout:如何用setTimeout连续打印0~9
    1. for(var i = 0; i < 10; i++) {//不使用bind
    2. (function(i) {
    3. setTimeout(function() {
    4. console.log(i);
    5. }, i * 1000);
    6. })(i);
    7. }
    8. for(var i = 0; i < 10; i++) {//使用bind
    9. setTimeout(console.log.bind(null, i), i * 1000);//给了console.log一个默认的参数i
    10. }

apply()、call()

参考文档:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Function/apply

  • 定义:

apply() 方法调用一个函数, 其具有一个指定的this值,以及作为一个数组(或类似数组的对象)提供的参数。
call() 方法调用一个函数, 其具有一个指定的this值,以及分别地提供的参数(参数的列表)。

  • 用法:

fun.apply(thisArg, [argsArray])
fun.call(thisArg, arg1,arg2,arg3,...)

  • 说明:

fun调用apply方法是这样fun里面的this指针指向thisArg,[argsArray]是一个数组或是类数组(arguments)作为fun函数的参数;
call方法除了参数和apply不一样其他都相同。

  • 示例:
  1. 使用call/apply实现继承:

    1. function a1(b){
    2. this.b = b;
    3. this.c = function(){
    4. console.log(this.b);
    5. }
    6. }
    7. function a2(b){
    8. a1.apply(this,[b]);
    9. //a1.call(this,b);
    10. }
    11.  
    12. var a3 = new a2(123);
    13. a3.c();//打印出123
  2. 使用apply和内置函数
    1. //得到最大值、最小值
    2. var arr = [5,6,9,11,0,1,3];
    3. var max = Math.max.apply(null,arr);
    4. var min = Math.min.apply(null,arr);
    5. console.log(max,min)//得到11,0

bind()、aplly()和call()的区别:

  • 返回值不同:调用bind()返回一个新的函数,aplly()和call()返回调用这两个方法的函数的返回值,如:
  1. function add(a,b){
  2. return a + b;
  3. }
  4. function sub(a,b){
  5. return a - b;
  6. }
  7. var a1 = add.bind(sub,3,2);
  8. var a2 = add.apply(sub,[3,2]);
  9. var a3 = sub.call(add,3,2);
  10. console.log(a1,a1(),a2,a3);//得到的是:ƒ add(a,b){return a + b;},5,5,1
  • 参数不同:bind()和call()的参数结构相同,都是this的新指向和一串参数,apply()的参数是this的新指向加一个数组或是类数组;

javascript中bind()、call()、apply()的使用的更多相关文章

  1. 面试官:能解释一下javascript中bind、apply和call这三个函数的用法吗

    一.前言    不知道大家还记不记得前几篇的文章:<面试官:能解释一下javascript中的this吗> 那今天这篇文章虽然是介绍javascript中bind.apply和call函数 ...

  2. 博文推荐】Javascript中bind、call、apply函数用法

    [博文推荐]Javascript中bind.call.apply函数用法 2015-03-02 09:22 菜鸟浮出水 51CTO博客 字号:T | T 最近一直在用 js 写游戏服务器,我也接触 j ...

  3. JavaScript函数 bind call apply区别

    1. apply calll 在JavaScript中 call 和 apply 都是为了改变某个函数运行时上下文而存在的, 换句话说就是为了改变函数内部的this的指向. 这里我们有一个新的对象 b ...

  4. JavaScript中call、apply个人理解

    JavaScript中call.apply个人理解 一句话即通俗的说:call.apply 是为了改变this的状态而存在的 }; } function personInfo(name,age){ t ...

  5. 说说 JavaScript中 call和apply

    下面有关JavaScript中 call和apply的描述,错误的是? call与apply都属于Function.prototype的一个方法,所以每个function实例都有call.apply属 ...

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

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

  7. Javascript中call、apply、bind函数

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

  8. Javascript中call,apply,bind的区别

    一.探索call方法原理 Function.prototype.call = function(obj) { // 1.让fn中的this指向obj // eval(this.toString().r ...

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

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

随机推荐

  1. [redis]带密码的客户端连接方法

    客户端连接方法: redis-cli -h localhost -p 6380 提供host为localhost,端口为6380   带密码的客户端连接方法一: redis-cli -h localh ...

  2. 品优购商城项目(二)AngularJS、自动代码生成器、select2下拉多选框

    品优购商城想项目第二阶段 AngularJS.自动代码生成器.select2下拉多选框 完成了课程第三天.第四天的的任务. 1.学习了AngularJs前端的mvc分层思想,js部分分成control ...

  3. QThread: Destroyed while thread is still running

    Qt5已经分装了函数 void QThread::requestInterruption() { Q_D(QThread); QMutexLocker locker(&d->mutex) ...

  4. 中标麒麟+Qt+mysql解决驱动无法加载的问题

    问题描述:都安装了Qt,Mysql之后,发现Qt始终不能连接Mysql 1.安装Qt 2.写程序直接连接QMysql 打印QSqlDatabase: * driver not loaded ,进入/h ...

  5. 【linux命令之 tail学习】

    tail 在屏幕上显示指定文件的末尾若干行 tail file #(显示文件file的最后10行) tail -n +20 file #(显示文件file的内容,从第20行至文件末尾) tail -c ...

  6. Linux下手动查杀木马

    (1).模拟木马程序病原体并让其自动运行 黑客让脚本自动执行的3种方法:1.计划任务:2.开机启动:3.系统命令被人替换,定一个触发事件. 1)生成木马程序病原体 [root@youxi1 ~]# v ...

  7. 毫无PS痕迹 你的第一本Photoshop书 完整版

    毫无PS痕迹 你的第一本Photoshop书 目录 <毫无PS痕迹-你的本Photoshop书>全书分为四大部分: 第1.2章讲解色彩和图像的原理与基础知识要点. 第3至11章全面讲解了使 ...

  8. Python subprocess中的run方法

    调用subprocess的推荐方法是对于它可以处理的所有使用场景都使用run()函数. run()函数是在Python 3.5中添加的,如果在老版本中使用,需要下载并扩展. 扩展安装方式: $ pip ...

  9. mybatis typeHandler类型转换器

    typeHandler类型转换器 在JDBC中,需要在PreparedStatement对象中设置那些已经预编译过的SQL语句的参数.执行SQL后,会通过ResultSet对象获取得到数据库的数据,而 ...

  10. bladex调用网关使用oauth2统一获取授权

    一:启动网关服务(需要启动其它服务,如:AuthApplication .UserApplication.BladeLogApplication,及自己添加的代码服务) 二:http://localh ...