浅谈JavaScript中的apply/call/bind和this的使用


Posted in Javascript onFebruary 26, 2017

fun.apply(context,[argsArray])

立即调用fun,同时将fun函数原来的this指向传入的新context对象,实现同一个方法在不同对象上重复使用。

context:传入的对象,替代fun函数原来的this;

argsArray:一个数组或者类数组对象,其中的数组参数会被展开作为单独的实参传给 fun 函数,需要注意参数的顺序。

fun.call(context,[arg1],[arg2],[…])

同apply,只是参数列表不同,call的参数需要分开一个一个传入。如果不知道参数个数,则使用apply。

使用:

Math.max()    //只接收单独的参数,通过下面的方法可以在数组上面使用max方法:
Math.max.apply(null, array);    //会将array数组参数展开成单独的参数再传入
Array.prototype.push.apply(arr1,arr2);    //将一个数组拆开push到另一个数组中;不用apply则会将后续数组参数当成一个元素push进去。
Array.prototype.slice.call(arguments);    //在类素组对象上使用slice方法

function isArray(obj){
  return Object.prototype.toString.call(obj) === '[object Array]' ;
}  //验证是否是数组

fun.bind(context,[arg1],[arg2],[…])

使fun方法执行的context永不变。

arg1:要传递到新函数的参数列表

返回一个函数供后续调用,其函数体和原函数fun一样,但新函数的this指向新传入的context对象。新函数会具有bind方法指定的初始参数arg1/arg2...,后续调用新函数时的实参要往已有参数的后面排。

//原来的函数有4个参数
var displayArgs = function (val1, val2, val3, val4) {
  console.log(val1 + " " + val2 + " " + val3 + " " + val4);
}
var emptyObject = {};
// 生成新函数时bind方法指定了2个参数,则新函数会带着这个两个实参
var displayArgs2 = displayArgs.bind(emptyObject, 12, "a");
// 调用时传入另2个参数,要在bind方法传入的2个实参后面
displayArgs2("b", "c");
// Output: 12 a b c

事件处理函数中使用bind:

var obj = {
  arg1 : 1,
  attach: function(){
    //var self = this; 普通传入this 的方法
    $('xxx').on('click',function (event) {
      console.log(this.arg1);//若不绑定this,回调函数中的this常指目标元素
     }.bind(this));  //使用bind方法绑定this
  }
}

使用bind()方法改写slice()方法:

var _Slice = Array.prototype.slice;
var slice = Function.prototype.call.bind(_Slice);
slice(…);

bind()兼容Ie5~ie8处理

if (!Function.prototype.bind) {
  Function.prototype.bind = function(context) {
    var self = this, // 调用bind方法的目标函数
    args = arguments;
    return function() {
      self.apply(context, Array.prototype.slice.call(args, 1));//参数个数不确定时用apply
    }
  }
}

一般情况下setTimeout()的this指向window或global对象。当使用类的方法时需要this指向类实例,就可以使用bind()将this绑定到调用对象,而不用传入self方式传入this。

this

this对象是在函数运行时基于函数的执行环境绑定的:在全局函数中,this等于window,而当函数被当作某个对象的方法调用时,this等于那个对象。

判断方法:this和定义在哪儿无关,函数运行时,如果有. 运算符,this指.前的对象;如果没有,this指window。若new关键字调用时,指代新对象。有apply/call/bind时,指代第一个参数。

/*例1*/
function foo() {
  console.log( this.a );
} 
var obj2 = {
  a: 42,
  foo: foo
};
var obj1 = {
  a: 2,
  obj2: obj2
};
obj1.obj2.foo(); // 42;当foo函数被调用时,其本身是归obj2所拥有
/*例2*/
function foo() {
  console.log( this.a );
} 
var obj = {
  a: 2,
  foo: foo
};
var bar = obj.foo;   // bar引用foo函数本身
var a = "global";   // 全局对象的属性
bar();        // "global" ;

在一个HTML DOM事件处理程序里面,this始终指向这个处理程序被所绑定到的DOM节点。

Javascript 相关文章推荐
原创javascript小游戏实现代码
Aug 19 Javascript
JavaScript初学者应注意的七个细节详细介绍
Dec 27 Javascript
jQuery的each终止或跳过示例代码
Dec 12 Javascript
JQuery中层次选择器用法实例详解
May 18 Javascript
浅谈js中调用函数时加不加括号的问题
Jul 28 Javascript
js获取指定字符前/后的字符串简单实例
Oct 27 Javascript
jquery结合html实现中英文页面切换
Nov 29 Javascript
JavaScript cookie详解及简单实例应用
Dec 31 Javascript
老生常谈js中的MVC
Jul 25 Javascript
详解a++和++a的区别
Aug 30 Javascript
微信小程序实现循环动画效果
Jul 16 Javascript
jQuery删除/清空指定元素的所有子节点实例代码
Jul 04 jQuery
JavaScript中Promise的使用详解
Feb 26 #Javascript
setTimeout函数的神奇使用
Feb 26 #Javascript
node.js入门学习之url模块
Feb 25 #Javascript
从零学习node.js之利用express搭建简易论坛(七)
Feb 25 #Javascript
从零学习node.js之express入门(六)
Feb 25 #Javascript
Node.JS中事件轮询(Event Loop)的解析
Feb 25 #Javascript
走进javascript——不起眼的基础,值和分号
Feb 24 #Javascript
You might like
PHP 5.0对象模型深度探索之对象复制
2008/03/27 PHP
PHP安全防范技巧分享
2011/11/03 PHP
php版微信公众平台回复中文出现乱码问题的解决方法
2016/09/22 PHP
浅析php中array_map和array_walk的使用对比
2016/11/20 PHP
ThinkPHP3.2框架自定义配置和加载用法示例
2018/06/14 PHP
PHP+Redis开发的书签案例实战详解
2019/07/09 PHP
LazyLoad 延迟加载(按需加载)
2010/05/31 Javascript
JavaScript实现点击按钮字体放大、缩小
2016/02/29 Javascript
浅谈JS读取DOM对象(标签)的自定义属性
2016/11/21 Javascript
基于js中的原型(全面讲解)
2017/09/19 Javascript
JS判断数组那点事
2017/10/10 Javascript
原生JS实现 MUI导航栏透明渐变效果
2017/11/07 Javascript
React Native之prop-types进行属性确认详解
2017/12/19 Javascript
video.js 实现视频只能后退不能快进的思路详解
2018/08/09 Javascript
如何手动实现es5中的bind方法详解
2018/12/07 Javascript
Python使用pygame模块编写俄罗斯方块游戏的代码实例
2015/12/08 Python
python数据预处理之将类别数据转换为数值的方法
2017/07/05 Python
阿里云ECS服务器部署django的方法
2019/08/29 Python
Python 解码Base64 得到码流格式文本实例
2020/01/09 Python
java字符串格式化输出实例讲解
2021/01/06 Python
Python实现微信表情包炸群功能
2021/01/28 Python
Python调用SMTP服务自动发送Email的实现步骤
2021/02/07 Python
wordpress添加Html5的表单验证required方法小结
2020/08/18 HTML / CSS
日本最大的药妆连锁店:Matsukiyo松本清药妆店
2017/11/23 全球购物
运动会广播稿400字
2014/01/25 职场文书
旷课检讨书1000字
2014/02/14 职场文书
指导教师评语
2014/04/26 职场文书
学校春季防火方案
2014/06/08 职场文书
七一建党日演讲稿
2014/09/05 职场文书
煤矿百日安全活动总结
2015/05/07 职场文书
2019学子的答谢词范本!
2019/07/05 职场文书
Python趣味爬虫之用Python实现智慧校园一键评教
2021/05/28 Python
css3带你实现3D转换效果
2022/02/24 HTML / CSS
口袋妖怪冰系十大最强精灵,几何雪花排第七,第六类似北极熊
2022/03/18 日漫
一次Mysql update sql不当引起的生产故障记录
2022/04/01 MySQL
instantclient客户端 连接oracle数据库
2022/04/26 Oracle