js的call apply bind 方法都很常见,目的都是为了改变某个方法的执行环境(context)

call

call([thisObj[,arg1[, arg2[,   [,.argN]]]]])

thisObj
可选项。将被用作当前对象的对象。
arg1, arg2, argN ..
可选项。将被传递方法参数序列。

如果没设置严格模式 “use strict”

  当thisObj 不存在或 为 undefined 或为 null 或为 this 时,则隐式地指向 全局对象(在浏览器中即为 window)

第二个参数是一个个值

apply

apply([thisObj[,arg1, arg2, argN]])

apply和call类似,区别只是第二个参数,是一个数组(或类数组)的形式

bind

bind(thisArg [, arg1 [, arg2, …]]);

bind 也是改变某个方法的执行环境,区别也在于第二个参数(也是一个个的参数形式)和“返回值”的特性。

  它将一个func绑定给thisArg的上下文,并传入相应的参数,并以一个新函数的形式返回,以供调用。

如 func.call(func1,var1,var2,var3)

对应的apply写法为:func.apply(func1,[var1,var2,var3])

对应的bind写法为: func.bind(func1,var1,var2,var3)()

来举个栗子:

//'use strict'

var name = 'name1';
var obj = {
name: 'name2',
sayName: function(str1,str2){
str1 = str1 || '';
str2 = str2 || '';
console.log(str1 + this.name + str2);
}
}; obj.sayName(); obj.sayName.bind(window,'Hello: ',' !')(); obj.sayName.apply(this,['hello: ',' ,']); obj.sayName.call(obj,'hello: ',' .');

将会输出:

注1:但IE9(包括IE9)以上的才支持bind

所以,在不支持bind的浏览器上,我们需要模拟一下

Function.prototype.Bind = function(context){
var self = this,
// 获取到bind第二个参数(中的所有参数)
args = Array.prototype.slice.call(arguments,1);
// 返回一个新的函数
return function(){
// 将相关参数赋给这个bind所在方法,并将执环境赋给context
return self.apply(context,args);
};
};

注2:

Function.prototype的apply和call是在1999年发布的ECMA262 Edition3中才加入的(1998年发布ECMA262 Edition2)。

在此前的的浏览器如IE5.01(JScript 5.0)中是没有apply和call的。因此也会带来一些兼容性问题。所以,

call的模拟:

Function.prototype.Call = function(context){
// 首先判断所给的context,即call的第一个参数
context = (context == undefined) ? window : context;
var temp = [],
evalStr = '';
// 最后要形成 一个eval字符串函数调用形式,以供动态执行
for(var i=1,j=arguments.length; i<j; i++){
temp.push('arguments[' + i + ']');
}
// 给context新增一个方法(拥有this值)
context._apply = this;
evalStr = 'context._apply(' + temp.join(',') + ')';
// console.log(evalStr);
try{
// 执行函数调用
eval(evalStr);
}catch(e){
throw new Error(e.message);
}finally{
// 销毁该属性
delete obj._apply;
}
};

apply的模拟:

apply也类似,因为第二个参数是类数组的形式,所以也要变换为数组

// 第二个参数 args是为了方便使用
Function.prototype.Apply = function(context,args){
context = (context == undefined) ? window : context;
var temp = [],
evalStr = '';
// 直接拿第二个参数数组的各个元素再进行组合join(',')
// 为什么不直接用 arguments[1]呢?
// 因为此时join也要用到 Array.prototype.join.call ,call又不一定支持
for(var i=0,j=args.length; i<j; i++){
temp.push('args[' + i + ']');
} context._apply = this;
evalStr = 'context._apply(' + temp.join(',') + ')';
// console.log(evalStr);
try{
eval(evalStr);
}catch(e){
throw new Error(e.message);
}finally{
delete obj._apply;
}
};

ok 来看一下对比效果

Function.prototype.Bind = function(context){
var self = this,
args = Array.prototype.slice.call(arguments,1);
return function(){
return self.apply(context,args);
};
}; Function.prototype.Call = function(context){
context = (context == undefined) ? window : context;
var temp = [],
evalStr = '';
for(var i=1,j=arguments.length; i<j; i++){
temp.push('arguments[' + i + ']');
}
context._apply = this;
evalStr = 'context._apply(' + temp.join(',') + ')';
console.log(evalStr);
try{
eval(evalStr);
}catch(e){
throw new Error(e.message);
}finally{
delete obj._apply;
}
}; Function.prototype.Apply = function(context,args){
context = (context == undefined) ? window : context;
var temp = [],
evalStr = '';
for(var i=0,j=args.length; i<j; i++){
temp.push('args[' + i + ']');
} context._apply = this;
evalStr = 'context._apply(' + temp.join(',') + ')';
console.log(evalStr);
try{
eval(evalStr);
}catch(e){
throw new Error(e.message);
}finally{
delete obj._apply;
}
}; var name = 'name1';
var obj = {
name: 'name2',
sayName: function(str1,str2){
str1 = str1 || '';
str2 = str2 || '';
console.log(str1 + this.name + str2);
}
}; obj.sayName(); obj.sayName.bind(window,'Hello: ',' !')();
obj.sayName.Bind(window,'Hello: ',' !')(); obj.sayName.apply(this,['hello: ',' ,']);
obj.sayName.Apply(this,['hello: ',' ,']); obj.sayName.call(obj,'hello: ',' .');
obj.sayName.Call(obj,'hello: ',' .');

JS 的 call apply bind 方法的更多相关文章

  1. JS中call,apply,bind方法的总结

    why?call,apply,bind干什么的?为什么要学这个? 一般用来指定this的环境,在没有学之前,通常会有这些问题. var a = { user: "小马扎", fn: ...

  2. JavaScript中call,apply,bind方法的区别

    call,apply,bind方法一般用来指定this的环境. var a = { user:"hahaha", fn:function(){ console.log(this.u ...

  3. JavaScript内置一些方法的实现原理--new关键字,call/apply/bind方法--前戏

    new关键字,call/apply/bind方法都和this的绑定有关,在学习之前,首先要理解this. 一起来学习一下this吧 首先.this是一个对象. 对象很好理解,引用类型值,可以实现如th ...

  4. JavaScript中call,apply,bind方法的总结。

    why?call,apply,bind干什么的?为什么要学这个? 一般用来指定this的环境,在没有学之前,通常会有这些问题. var a = { user:"追梦子", fn:f ...

  5. js: this,call,apply,bind 总结

    对js中的一些基本的很重要的概念做一些总结,对进一步学习js很重. 一.this JavaScript 中的 this 总是指向一个对象,而具体指向那个对象是在运行时基于函数的执行环境动态绑定的,而非 ...

  6. call,apply,bind方法的总结

    why?call,apply,bind干什么的?为什么要学这个? 一般用来指定this的环境,在没有学之前,通常会有这些问题. var a = { user:"追梦子", fn:f ...

  7. JavaScript中call,apply,bind方法的总结

    原文链接:http://www.cnblogs.com/pssp/p/5215621.html why?call,apply,bind干什么的?为什么要学这个? 一般用来指定this的环境,在没有学之 ...

  8. JavaScript中call,apply,bind方法

    why?call,apply,bind干什么的?为什么要学这个? 一般用来指定this的环境,在没有学之前,通常会有这些问题. var a = { user:"追梦子", fn:f ...

  9. prototype.js中Function.prototype.bind方法浅解

    prototype.js中的Function.prototype.bind方法: Function.prototype.bind = function() { var __method = this; ...

随机推荐

  1. Md5加密方法

    package com.atguigu.surveypark.util; import java.security.MessageDigest; /** * 数据 */ public class Da ...

  2. sublime3 配置node build环境

    折腾了很久,原来如此简单 1.package control  安装nodejs 2.修改Nodejs.sublime-settings文件,将nodejs路径修改成自己的 3.sublime tex ...

  3. [数据库]SQL Server 用户NT AUTHORITY\IUSR 登录失败

    今天打开网站时,突然报这个错误,平时都好好的 Cannot open database "JMECC" requested by the login. The login fail ...

  4. python 字符编码问题

    原因 近期,用Python处理一些中文的字符串,但是用Python读取文件的中文字符和在代码中输入的中文字符,在判断处理时,例如判断“相等”或者“包含”,总是不能判断,相等或者包含都不起作用.看了字符 ...

  5. centos7 Linux 尝试使用crontab

    一.安装crontab [root@CentOS ~]# yum install vixie-cron[root@CentOS ~]# yum install crontabs 说明:vixie-cr ...

  6. Django 源码小剖: 应用程序入口 WSGIHandler

    WSGI 有三个部分, 分别为服务器(server), 应用程序(application) 和中间件(middleware). 已经知道, 服务器方面会调用应用程序来处理请求, 在应用程序中有真正的处 ...

  7. GPS accuracy in Android

    Get the estimated accuracy of this location, in meters. We define accuracy as the radius of 68% conf ...

  8. iOS 实现快速切换主题详细教程(附上源码)

    前言 iOS 实现主题切换,相信在未来的app里也是会频繁出现的,尽管现在只是出现在主流的APP,如(QQ.新浪微博.酷狗音乐.网易云音乐等),但是现在是看颜值.追求个性的年代,所以根据用户喜好自定义 ...

  9. 做最好的日期控件,My97 DatePicker 4.8 Beta4

    4.8更新的内容 [新增]preload预载选项 [新增]时分秒选择菜单的定制功能,详见hmsMenuCfg属性[beta4] [新增]$dp.unbind函数,允许用户手动接触日期控件绑定[beta ...

  10. -webkit-min-device-pixel-ratio的常见值对照

    -webkit-min-device-pixel-ratio为1.0 所有非Retina的Mac 所有非Retina的iOS设备 Acer Iconia A500 Samsung Galaxy Tab ...