不用call和apply方法模拟实现ES5的bind方法
本文首发我的个人博客:前端小密圈,评论交流送1024邀请码,嘿嘿嘿😄。
来自朋友去某信用卡管家的做的一道面试题,用原生JavaScript
模拟ES5
的bind
方法,不准用call
和bind
方法。
至于结果嘛。。。那个人当然是没写出来,我就自己尝试研究了一番,其实早就写了,一直没有组织好语言发出来。
额。。。这个题有点***钻,这是对JavaScript
基本功很好的一个检测,看你JavaScript
掌握的怎么样以及平时有没有去深入研究一些方法的实现,简而言之,就是有没有折腾精神。
不准用不用call
和apply
方法,这个没啥好说的,不准用我们就用原生JavaScript
先来模拟一个apply
方法,感兴趣的童鞋也可以看看chrome
的v8
怎么实现这个方法的,这里我只按照自己的思维实现,在模拟之前我们先要明白和了解原生call
和apply
方法是什么。
简单粗暴地来说,call
,apply
,bind
是用于绑定this
指向的。(如果你还不了解JS中this的指向问题,以及执行环境上下文的奥秘,这篇文章暂时就不太适合阅读)。
什么是call和apply方法
我们单独看看ECMAScript
规范对apply
的定义,看个大概就行:
15.3.4.3 Function.prototype.apply (thisArg, argArray)
顺便贴一贴中文版,免得翻译一下,中文版地址:
通过定义简单说一下call和apply方法,他们就是参数不同,作用基本相同。
1、每个函数都包含两个非继承而来的方法:apply()和call()。
2、他们的用途相同,都是在特定的作用域中调用函数。
3、接收参数方面不同,apply()接收两个参数,一个是函数运行的作用域(this),另一个是参数数组。
4、call()方法第一个参数与apply()方法相同,但传递给函数的参数必须列举出来。
知道定义然后,直接看个简单的demo
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
var jawil = {
name: "jawil",
sayHello: function (age) {
console.log("hello, i am ", this.name + " " + age + " years old");
}
};
var lulin = {
name: "lulin",
};
jawil.sayHello(24);
// hello, i am jawil 24 years old
|
然后看看使用apply
和call
之后的输出:
1
2
3
|
jawil.sayHello.call(lulin, 24);// hello, i am lulin 24 years old
jawil.sayHello.apply(lulin, [24]);// hello, i am lulin 24 years old
|
结果都相同。从写法上我们就能看出二者之间的异同。相同之处在于,第一个参数都是要绑定的上下文,后面的参数是要传递给调用该方法的函数的。不同之处在于,call方法传递给调用函数的参数是逐个列出的,而apply则是要写在数组中。
总结一句话介绍call
和apply
call()
方法在使用一个指定的this
值和若干个指定的参数值的前提下调用某个函数或方法。apply()
方法在使用一个指定的this
值和参数值必须是数组类型的前提下调用某个函数或方法。
分析call和apply的原理
上面代码,我们注意到了两点:
call
和apply
改变了this
的指向,指向到lulin
sayHello
函数执行了
这里默认大家都对this
有一个基本的了解,知道什么时候this
该指向谁,我们结合这两句话来分析这个通用函数:f.apply(o)
,我们直接看一本书对其中原理的解读,具体什么书,我也不知道,参数我们先不管,先了解其中的大致原理。
注意红色框中的部分,f.call(o)其原理就是先通过 o.m = f 将 f作为o的某个临时属性m存储,然后执行m,执行完毕后将m属性删除。
知道了这个基本原来我们再来看看刚才jawil.sayHello.call(lulin, 24)
执行的过程:
1
2
3
4
5
6
|
// 第一步
lulin.fn = jawil.sayHello
// 第二步
lulin.fn()
// 第三步
delete lulin.fn
|
上面的说的是原理,可能你看的还有点抽象,下面我们用代码模拟实现apply
一下。
实现aplly方法
模拟实现第一步
根据这个思路,我们可以尝试着去写第一版的 applyOne 函数:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
// 第一版
Function.prototype.applyOne = function(context) {
// 首先要获取调用call的函数,用this可以获取
context.fn = this;
context.fn();
delete context.fn;
}
//简单写一个不带参数的demo
var jawil = {
name: "jawil",
sayHello: function (age) {
console.log(this.name);
}
};
var lulin = {
name: "lulin",
};
//看看结果:
jawil.sayHello.applyOne(lulin)//lulin
|
正好可以打印lulin而不是之前的jawil了,哎,不容易啊!😄
模拟实现第二步
最一开始也讲了,apply
函数还能给定参数执行函数。举个例子:
1
2
3
4
5
6
7
8
9
10
11
12
|
var jawil = {
name: "jawil",
sayHello: function (age) {
console.log(this.name,age);
}
};
var lulin = {
name: "lulin",
};
jawil.sayHello.apply(lulin,[24])//lulin 24
|
注意:传入的参数就是一个数组,很简单,我们可以从Arguments
对象中取值,Arguments
不知道是何物,赶紧补习,此文也不太适合初学者,第二个参数就是数组对象,但是执行的时候要把数组数值传递给函数当参数,然后执行,这就需要一点小技巧。
参数问题其实很简单,我们先偷个懒,我们接着要把这个参数数组放到要执行的函数的参数里面去。
1
2
3
4
5
6
7
|
Function.prototype.applyTwo = function(context) {
// 首先要获取调用call的函数,用this可以获取
context.fn = this;
var args = arguments[1] //获取传入的数组参数
context.fn(args.join(',');
delete context.fn;
}
|
很简单是不是,那你就错了,数组join方法返回的是啥?
typeof [1,2,3,4].join(',')//string
Too young,too simple啊,最后是一个 “1,2,3,4” 的字符串,其实就是一个参数,肯定不行啦。
也许有人会想到用ES6的一些奇淫方法,不过apply
是ES3
的方法,我们为了模拟实现一个ES3
的方法,要用到ES6
的方法,反正面试官也没说不准这样。但是我们这次用eval
方法拼成一个函数,类似于这样:
eval('context.fn(' + args +')')
先简单了解一下eval函数吧
定义和用法
eval() 函数可计算某个字符串,并执行其中的的 JavaScript 代码。
语法:eval(string)
string必需。要计算的字符串,其中含有要计算的 JavaScript 表达式或要执行的语句。该方法只接受原始字符串作为参数,如果 string 参数不是原始字符串,那么该方法将不作任何改变地返回。因此请不要为 eval() 函数传递 String 对象来作为参数。
简单来说吧,就是用JavaScript的解析引擎来解析这一堆字符串里面的内容,这么说吧,你可以这么理解,你把eval
看成是<script>
标签。
eval('function Test(a,b,c,d){console.log(a,b,c,d)};Test(1,2,3,4)')
就是相当于这样
1
2
3
4
5
6
|
<script>
function Test(a,b,c,d){
console.log(a,b,c,d)
};
Test(1,2,3,4)
</script>
|
第二版代码大致如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
Function.prototype.applyTwo = function(context) {
var args = arguments[1]; //获取传入的数组参数
context.fn = this; //假想context对象预先不存在名为fn的属性
var fnStr = 'context.fn(';
for (var i = 0; i < args.length; i++) {
fnStr += i == args.length - 1 ? args[i] : args[i] + ',';
}
fnStr += ')';//得到"context.fn(arg1,arg2,arg3...)"这个字符串在,最后用eval执行
eval(fnStr); //还是eval强大
delete context.fn; //执行完毕之后删除这个属性
}
//测试一下
var jawil = {
name: "jawil",
sayHello: function (age) {
console.log(this.name,age);
}
};
var lulin = {
name: "lulin",
};
jawil.sayHello.applyTwo(lulin,[24])//lulin 24
|
好像就行了是不是,其实这只是最粗糙的版本,能用,但是不完善,完成了大约百分之六七十了。
模拟实现第三步
其实还有几个小地方需要注意:
1.this
参数可以传null
或者不传,当为null
的时候,视为指向window
举个两个简单栗子栗子🌰:
demo1:
1
2
3
4
5
6
7
|
var name = 'jawil';
function sayHello() {
console.log(this.name);
}
sayHello.apply(null); // 'jawil'
|
demo2:
1
2
3
4
5
6
7
|
var name = 'jawil';
function sayHello() {
console.log(this.name);
}
sayHello.apply(); // 'jawil'
|
2.函数是可以有返回值的.
举个简单栗子🌰:
1
2
3
4
5
6
7
8
9
10
11
12
|
var obj = {
name: 'jawil'
}
function sayHello(age) {
return {
name: this.name,
age: age
}
}
console.log(sayHello.apply(obj,[24]));// {name: "jawil", age: 24}
|
这些都是小问题,想到了,就很好解决。我们来看看此时的第三版apply
模拟方法。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
//原生JavaScript封装apply方法,第三版
Function.prototype.applyThree = function(context) {
var context = context || window
var args = arguments[1] //获取传入的数组参数
context.fn = this //假想context对象预先不存在名为fn的属性
if (args == void 0) { //没有传入参数直接执行
return context.fn()
}
var fnStr = 'context.fn('
for (var i = 0; i < args.length; i++) {
//得到"context.fn(arg1,arg2,arg3...)"这个字符串在,最后用eval执行
fnStr += i == args.length - 1 ? args[i] : args[i] + ','
}
fnStr += ')'
var returnValue = eval(fnStr) //还是eval强大
delete context.fn //执行完毕之后删除这个属性
return returnValue
}
|
好紧张,再来做个小测试,demo,应该不会出问题:
1
2
3
4
5
6
7
8
9
10
11
12
|
var obj = {
name: 'jawil'
}
function sayHello(age) {
return {
name: this.name,
age: age
}
}
console.log(sayHello.applyThree(obj,[24]));// 完美输出{name: "jawil", age: 24}
|
完美?perfact?这就好了,不存在的,我们来看看第四步的实现。
模拟实现第四步
其实一开始就埋下了一个隐患,我们看看这段代码:
1
2
3
4
5
6
|
Function.prototype.applyThree = function(context) {
var context = context || window
var args = arguments[1] //获取传入的数组参数
context.fn = this //假想context对象预先不存在名为fn的属性
......
}
|
就是这句话, context.fn = this //假想context对象预先不存在名为fn的属性
,这就是一开始的隐患,我们只是假设,但是并不能防止contenx
对象一开始就没有这个属性,要想做到完美,就要保证这个context.fn
中的fn
的唯一性。
于是我自然而然的想到了强大的ES6
,这玩意还是好用啊,幸好早就了解并一直在使用ES6
,还没有学习过ES6的童鞋赶紧学习一下,没有坏处的。
重新复习下新知识:
基本数据类型有6种:Undefined
、Null
、布尔值(Boolean)
、字符串(String)
、数值(Number)
、对象(Object)
。
ES5对象属性名都是字符串容易造成属性名的冲突。
举个栗子🌰:
1
2
3
|
var a = { name: 'jawil'};
a.name = 'lulin';
//这样就会重写属性
|
ES6
引入了一种新的原始数据类型Symbol
,表示独一无二的值。
注意,Symbol
函数前不能使用new
命令,否则会报错。这是因为生成的Symbol
是一个原始类型的值,不是对象
Symbol
函数可以接受一个字符串作为参数,表示对Symbol
实例的描述,主要是为了在控制台显示,或者转为字符串时,比较容易区分。
1
2
3
4
5
6
7
8
9
10
11
|
// 没有参数的情况
var s1 = Symbol();
var s2 = Symbol();
s1 === s2 // false
// 有参数的情况
var s1 = Symbol("foo");
var s2 = Symbol("foo");
s1 === s2 // false
|
注意:
Symbol
值不能与其他类型的值进行运算。
作为属性名的Symbol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
var mySymbol = Symbol();
// 第一种写法
var a = {};
a[mySymbol] = 'Hello!';
// 第二种写法
var a = {
[mySymbol]: 'Hello!'
};
// 第三种写法
var a = {};
Object.defineProperty(a, mySymbol, { value: 'Hello!' });
// 以上写法都得到同样结果
a[mySymbol] // "Hello!"
|
注意,Symbol值作为对象属性名时,不能用点运算符。
看看下面这个栗子🌰:
1
2
3
4
5
|
var a = {};
var name = Symbol();
a.name = 'jawil';
a[name] = 'lulin';
console.log(a.name,a[name]); //jawil,lulin
|
Symbol
值作为属性名时,该属性还是公开属性,不是私有属性。
这个有点类似于java
中的protected
属性(protected和private的区别:在类的外部都是不可以访问的,在类内的子类可以继承protected不可以继承private)
但是这里的Symbol在类外部也是可以访问的,只是不会出现在for...in
、for...of
循环中,也不会被Object.keys()
、Object.getOwnPropertyNames()
返回。但有一个Object.getOwnPropertySymbols
方法,可以获取指定对象的所有Symbol
属性名。
看看第四版的实现demo,想必大家了解上面知识已经猜得到怎么写了,很简单。
直接加个var fn = Symbol()
就行了,,,
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
//原生JavaScript封装apply方法,第四版
Function.prototype.applyFour = function(context) {
var context = context || window
var args = arguments[1] //获取传入的数组参数
var fn = Symbol()
context[fn] = this //假想context对象预先不存在名为fn的属性
if (args == void 0) { //没有传入参数直接执行
return context[fn]()
}
var fnStr = 'context[fn]('
for (var i = 0; i < args.length; i++) {
//得到"context.fn(arg1,arg2,arg3...)"这个字符串在,最后用eval执行
fnStr += i == args.length - 1 ? args[i] : args[i] + ','
}
fnStr += ')'
var returnValue = eval(fnStr) //还是eval强大
delete context[fn] //执行完毕之后删除这个属性
return returnValue
}
|
模拟实现第五步
呃呃呃额额,慢着,ES3
就出现的方法,你用ES6
来实现,你好意思么?你可能会说,不管黑猫白猫,只要能抓住老鼠的猫就是好猫,面试官直说不准用call
和apply
方法但是没说不准用ES6
语法啊。
反正公说公有理婆说婆有理,这里还是不用Symbol
方法实现一下,我们知道,ES6其实都是语法糖,ES6
能写的,咋们ES5
都能实现,这就导致了babel
这类把ES6
语法转化成ES5
的代码了。
至于babel
把Symbol
属性转换成啥代码了,我也没去看,有兴趣的可以看一下稍微研究一下,这里我说一下简单的模拟。
ES5
没有 Sybmol
,属性名称只可能是一个字符串,如果我们能做到这个字符串不可预料,那么就基本达到目标。要达到不可预期,一个随机数基本上就解决了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
//简单模拟Symbol属性
function jawilSymbol(obj) {
var unique_proper = "00" + Math.random();
if (obj.hasOwnProperty(unique_proper)) {
arguments.callee(obj)//如果obj已经有了这个属性,递归调用,直到没有这个属性
} else {
return unique_proper;
}
}
//原生JavaScript封装apply方法,第五版
Function.prototype.applyFive = function(context) {
var context = context || window
var args = arguments[1] //获取传入的数组参数
var fn = jawilSymbol(context);
context[fn] = this //假想context对象预先不存在名为fn的属性
if (args == void 0) { //没有传入参数直接执行
return context[fn]()
}
var fnStr = 'context[fn]('
for (var i = 0; i < args.length; i++) {
//得到"context.fn(arg1,arg2,arg3...)"这个字符串在,最后用eval执行
fnStr += i == args.length - 1 ? args[i] : args[i] + ','
}
fnStr += ')'
var returnValue = eval(fnStr) //还是eval强大
delete context[fn] //执行完毕之后删除这个属性
return returnValue
}
|
好紧张,再来做个小测试,demo,应该不会出问题:
1
2
3
4
5
6
7
8
9
10
11
12
|
var obj = {
name: 'jawil'
}
function sayHello(age) {
return {
name: this.name,
age: age
}
}
console.log(sayHello.applyFive(obj,[24]));// 完美输出{name: "jawil", age: 24}
|
到此,我们完成了apply的模拟实现,给自己一个赞 b( ̄▽ ̄)d
实现Call方法
这个不需要讲了吧,道理都一样,就是参数一样,这里我给出我实现的一种方式,看不懂,自己写一个去。
1
2
3
4
5
|
//原生JavaScript封装call方法
Function.prototype.callOne = function(context) {
return this.applyFive(([].shift.applyFive(arguments), arguments)
//巧妙地运用上面已经实现的applyFive函数
}
|
看不太明白也不能怪我咯,我就不细讲了,看个demo证明一下,这个写法没问题。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
Function.prototype.applyFive = function(context) {//刚才写的一大串}
Function.prototype.callOne = function(context) {
return this.applyFive(([].shift.applyFive(arguments)), arguments)
//巧妙地运用上面已经实现的applyFive函数
}
//测试一下
var obj = {
name: 'jawil'
}
function sayHello(age) {
return {
name: this.name,
age: age
}
}
console.log(sayHello.callOne(obj,24));// 完美输出{name: "jawil", age: 24}
|
实现bind方法
养兵千日,用兵一时。
什么是bind函数
如果掌握了上面实现apply
的方法,我想理解起来模拟实现bind
方法也是轻而易举,原理都差不多,我们还是来看看bind
方法的定义。
我们还是简单的看下ECMAScript
规范对bind
方法的定义,暂时看不懂不要紧,获取几个关键信息就行。
15.3.4.5 Function.prototype.bind (thisArg [, arg1 [, arg2, …]])
注意一点,ECMAScript规范提到: Function.prototype.bind 创建的函数对象不包含 prototype 属性或 [[Code]], [[FormalParameters]], [[Scope]] 内部属性。
bind() 方法会创建一个新函数,当这个新函数被调用时,它的
this
值是传递给bind()
的第一个参数, 它的参数是bind()
的其他参数和其原本的参数,bind返回的绑定函数也能使用new操作符创建对象:这种行为就像把原函数当成构造器。提供的this值被忽略,同时调用时的参数被提供给模拟函数。。
语法是这样样子的:fun.bind(thisArg[, arg1[, arg2[, ...]]])
呃呃呃,是不是似曾相识,这不是call方法的语法一个样子么,,,但它们是一样的吗?
bind方法传递给调用函数的参数可以逐个列出,也可以写在数组中。bind方法与call、apply最大的不同就是前者返回一个绑定上下文的函数,而后两者是直接执行了函数。由于这个原因,上面的代码也可以这样写:
1
2
|
jawil.sayHello.bind(lulin)(24); //hello, i am lulin 24 years old
jawil.sayHello.bind(lulin)([24]); //hello, i am lulin 24 years old
|
bind方法还可以这样写 fn.bind(obj, arg1)(arg2)
.
用一句话总结bind的用法:该方法创建一个新函数,称为绑定函数,绑定函数会以创建它时传入bind方法的第一个参数作为this,传入bind方法的第二个以及以后的参数加上绑定函数运行时本身的参数按照顺序作为原函数的参数来调用原函数。
bind在实际中的应用
实际使用中我们经常会碰到这样的问题:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
function Person(name){
this.nickname = name;
this.distractedGreeting = function() {
setTimeout(function(){
console.log("Hello, my name is " + this.nickname);
}, 500);
}
}
var alice = new Person('jawil');
alice.distractedGreeting();
//Hello, my name is undefined
|
这个时候输出的this.nickname是undefined,原因是this指向是在运行函数时确定的,而不是定义函数时候确定的,再因为setTimeout在全局环境下执行,所以this指向setTimeout的上下文:window。关于this指向问题,这里就不细扯
以前解决这个问题的办法通常是缓存this
,例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
function Person(name){
this.nickname = name;
this.distractedGreeting = function() {
var self = this; // <-- 注意这一行!
setTimeout(function(){
console.log("Hello, my name is " + self.nickname); // <-- 还有这一行!
}, 500);
}
}
var alice = new Person('jawil');
alice.distractedGreeting();
// after 500ms logs "Hello, my name is jawil"
|
这样就解决了这个问题,非常方便,因为它使得setTimeout函数中可以访问Person的上下文。但是看起来稍微一种蛋蛋的忧伤。
但是现在有一个更好的办法!您可以使用bind
。上面的例子中被更新为:
1
2
3
4
5
6
7
8
9
10
11
12
|
function Person(name){
this.nickname = name;
this.distractedGreeting = function() {
setTimeout(function(){
console.log("Hello, my name is " + this.nickname);
}.bind(this), 500); // <-- this line!
}
}
var alice = new Person('jawil');
alice.distractedGreeting();
// after 500ms logs "Hello, my name is jawil"
|
bind() 最简单的用法是创建一个函数,使这个函数不论怎么调用都有同样的 this 值。JavaScript新手经常犯的一个错误是将一个方法从对象中拿出来,然后再调用,希望方法中的 this 是原来的对象。(比如在回调中传入这个方法。)如果不做特殊处理的话,一般会丢失原来的对象。从原来的函数和原来的对象创建一个绑定函数,则能很漂亮地解决这个问题:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
this.x = 9;
var module = {
x: 81,
getX: function() { return this.x; }
};
module.getX(); // 81
var getX = module.getX;
getX(); // 9, 因为在这个例子中,"this"指向全局对象
// 创建一个'this'绑定到module的函数
var boundGetX = getX.bind(module);
boundGetX(); // 81
|
很不幸,Function.prototype.bind 在IE8及以下的版本中不被支持,所以如果你没有一个备用方案的话,可能在运行时会出现问题。bind 函数在 ECMA-262 第五版才被加入;它可能无法在所有浏览器上运行。你可以部份地在脚本开头加入以下代码,就能使它运作,让不支持的浏览器也能使用 bind() 功能。
幸运的是,我们可以自己来模拟bind
功能:
初级实现
了解了以上内容,我们来实现一个初级的bind
函数Polyfill
:
1
2
3
4
5
6
7
|
Function.prototype.bind = function (context) {
var me = this;
var argsArray = Array.prototype.slice.call(arguments);
return function () {
return me.apply(context, argsArray.slice(1))
}
}
|
我们先简要解读一下:
基本原理是使用apply
进行模拟。函数体内的this
,就是需要绑定this
的实例函数,或者说是原函数。最后我们使用apply
来进行参数(context)绑定,并返回。
同时,将第一个参数(context)以外的其他参数,作为提供给原函数的预设参数,这也是基本的“颗粒化(curring)”基础。
初级实现的加分项
上面的实现(包括后面的实现),其实是一个典型的“Monkey patching(猴子补丁)”,即“给内置对象扩展方法”。所以,如果面试者能进行一下“嗅探”,进行兼容处理,就是锦上添花了。
1
2
3
|
Function.prototype.bind = Function.prototype.bind || function (context) {
...
}
|
颗粒化(curring)实现
对于函数的柯里化不太了解的童鞋,可以先尝试读读这篇文章:前端基础进阶(八):深入详解函数的柯里化。
上述的实现方式中,我们返回的参数列表里包含:atgsArray.slice(1)
,他的问题在于存在预置参数功能丢失的现象。
想象我们返回的绑定函数中,如果想实现预设传参(就像bind
所实现的那样),就面临尴尬的局面。真正实现颗粒化的“完美方式”是:
1
2
3
4
5
6
7
8
9
|
Function.prototype.bind = Function.prototype.bind || function (context) {
var me = this;
var args = Array.prototype.slice.call(arguments, 1);
return function () {
var innerArgs = Array.prototype.slice.call(arguments);
var finalArgs = args.concat(innerArgs);
return me.apply(context, finalArgs);
}
}
|
上面什么是bind函数还介绍到:bind返回的函数如果作为构造函数,搭配new关键字出现的话,我们的绑定this就需要“被忽略”。
构造函数场景下的兼容
有了上边的讲解,不难理解需要兼容构造函数场景的实现:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
Function.prototype.bind = Function.prototype.bind || function (context) {
var me = this;
var args = Array.prototype.slice.call(arguments, 1);
var F = function () {};
F.prototype = this.prototype;
var bound = function () {
var innerArgs = Array.prototype.slice.call(arguments);
var finalArgs = args.concat(innerArgs);
return me.apply(this instanceof F ? this : context || this, finalArgs);
}
bound.prototype = new F();
return bound;
}
|
更严谨的做法
我们需要调用bind
方法的一定要是一个函数,所以可以在函数体内做一个判断:
1
2
3
|
if (typeof this !== "function") {
throw new TypeError("Function.prototype.bind - what is trying to be bound is not callable");
}
|
做到所有这一切,基本算是完成了。其实MDN上有个自己实现的polyfill,就是如此实现的。
另外,《JavaScript Web Application》一书中对bind()的实现,也是如此。
最终答案
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
|
//简单模拟Symbol属性
function jawilSymbol(obj) {
var unique_proper = "00" + Math.random();
if (obj.hasOwnProperty(unique_proper)) {
arguments.callee(obj)//如果obj已经有了这个属性,递归调用,直到没有这个属性
} else {
return unique_proper;
}
}
//原生JavaScript封装apply方法,第五版
Function.prototype.applyFive = function(context) {
var context = context || window
var args = arguments[1] //获取传入的数组参数
var fn = jawilSymbol(context);
context[fn] = this //假想context对象预先不存在名为fn的属性
if (args == void 0) { //没有传入参数直接执行
return context[fn]()
}
var fnStr = 'context[fn]('
for (var i = 0; i < args.length; i++) {
//得到"context.fn(arg1,arg2,arg3...)"这个字符串在,最后用eval执行
fnStr += i == args.length - 1 ? args[i] : args[i] + ','
}
fnStr += ')'
var returnValue = eval(fnStr) //还是eval强大
delete context[fn] //执行完毕之后删除这个属性
return returnValue
}
//简单模拟call函数
Function.prototype.callOne = function(context) {
return this.applyFive(([].shift.applyFive(arguments)), arguments)
//巧妙地运用上面已经实现的applyFive函数
}
//简单模拟bind函数
Function.prototype.bind = Function.prototype.bind || function (context) {
var me = this;
var args = Array.prototype.slice.callOne(arguments, 1);
var F = function () {};
F.prototype = this.prototype;
var bound = function () {
var innerArgs = Array.prototype.slice.call(arguments);
var finalArgs = args.concat(innerArgs);
return me.applyFive(this instanceof F ? this : context || this, finalArgs);
}
bound.prototype = new F();
return bound;
}
|
好紧张,最后来做个小测试,demo,应该不会出问题:
1
2
3
4
5
6
7
8
9
10
11
12
|
var obj = {
name: 'jawil'
}
function sayHello(age) {
return {
name: this.name,
age: age
}
}
console.log(sayHello.bind(obj,24)());// 完美输出{name: "jawil", age: 24}
|
看了这篇文章,以后再遇到类似的问题,应该能够顺利通过吧~