You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
// 看看V8引擎中的具体实现:
function ArrayPush() {
var n = TO_UINT32( this.length ); // 被push的对象的length
var m = %_ArgumentsLength(); // push的参数个数
for (var i = 0; i < m; i++) {
this[ i + n ] = %_Arguments( i ); // 复制元素 (1)
}
this.length = n + m; // 修正length属性的值 (2)
return this.length;
};
之前写过两篇《面试官问:能否模拟实现
JS
的new
操作符》和《面试官问:能否模拟实现JS
的bind
方法》其中模拟
bind
方法时是使用的call
和apply
修改this
指向。但面试官可能问:能否不用call
和apply
来实现呢。意思也就是需要模拟实现call
和apply
的了。先通过
MDN
认识下call
和apply
MDN 文档:Function.prototype.call()
语法
thisArg
在
fun
函数运行时指定的this
值。需要注意的是,指定的this
值并不一定是该函数执行时真正的this
值,如果这个函数处于非严格模式下,则指定为null
和undefined
的this
值会自动指向全局对象(浏览器中就是window
对象),同时值为原始值(数字,字符串,布尔值)的this
会指向该原始值的自动包装对象。arg1, arg2, ...
指定的参数列表
返回值
返回值是你调用的方法的返回值,若该方法没有返回值,则返回
undefined
。MDN 文档:Function.prototype.apply()
thisArg
可选的。在
func
函数运行时使用的this
值。请注意,this
可能不是该方法看到的实际值:如果这个函数处于非严格模式下,则指定为null
或undefined
时会自动替换为指向全局对象,原始值会被包装。argsArray
可选的。一个数组或者类数组对象,其中的数组元素将作为单独的参数传给
func
函数。如果该参数的值为null
或undefined
,则表示不需要传入任何参数。从ECMAScript 5
开始可以使用类数组对象。返回值
调用有指定this值和参数的函数的结果。
直接先看例子1
call
和apply
的异同相同点:
1、
call
和apply
的第一个参数thisArg
,都是func
运行时指定的this
。而且,this
可能不是该方法看到的实际值:如果这个函数处于非严格模式下,则指定为null
或undefined
时会自动替换为指向全局对象,原始值会被包装。2、都可以只传递一个参数。
不同点:
apply
只接收两个参数,第二个参数可以是数组也可以是类数组,其实也可以是对象,后续的参数忽略不计。call
接收第二个及以后一系列的参数。看两个简单例子1和2**:
typeof
有7
种类型(undefined number string boolean symbol object function
),笔者都验证了一遍:更加验证了相同点第一点,严格模式下,函数的this
值就是call
和apply
的第一个参数thisArg
,非严格模式下,thisArg
值被指定为null
或undefined
时this
值会自动替换为指向全局对象,原始值则会被自动包装,也就是new Object()
。重新认识了
call
和apply
会发现:它们作用都是一样的,改变函数里的this
指向为第一个参数thisArg
,如果明确有多少参数,那可以用call
,不明确则可以使用apply
。也就是说完全可以不使用call
,而使用apply
代替。也就是说,我们只需要模拟实现
apply
,call
可以根据参数个数都放在一个数组中,给到apply
即可。模拟实现
apply
既然准备模拟实现
apply
,那先得看看ES5
规范。ES5规范 英文版
,ES5规范 中文版
。apply
的规范下一个就是call
的规范,可以点击打开新标签页去查看,这里摘抄一部分。结合上文和规范,如何将函数里的
this
指向第一个参数thisArg
呢,这是一个问题。这时候请出例子3:
可以得出结论1:在对象
student
上加一个函数doSth
,再执行这个函数,这个函数里的this
就指向了这个对象。那也就是可以在thisArg
上新增调用函数,执行后删除这个函数即可。知道这些后,我们试着容易实现第一版本:
实现第一版后,很容易找出两个问题:
__fn
同名覆盖问题,thisArg
对象上有__fn
,那就被覆盖了然后被删除了。针对问题1
解决方案一:采用
ES6
Sybmol()
独一无二的。可以本来就是模拟ES3
的方法。如果面试官不允许用呢。解决方案二:自己用
Math.random()
模拟实现独一无二的key
。面试时可以直接用生成时间戳即可。如果这个
key
万一这对象中还是有,为了保险起见,可以做一次缓存操作。比如如下代码:ES6
扩展符...
解决方案一:采用
eval
来执行函数。参数
string
表示
JavaScript
表达式,语句或一系列语句的字符串。表达式可以包含变量以及已存在对象的属性。返回值
执行指定代码之后的返回值。如果返回值为空,返回
undefined
解决方案二:但万一面试官不允许用
eval
呢,毕竟eval
是魔鬼。可以采用new Function()
来生成执行函数。MDN 文档:Function
语法
参数
arg1, arg2, ... argN
被函数使用的参数的名称必须是合法命名的。参数名称是一个有效的
JavaScript
标识符的字符串,或者一个用逗号分隔的有效字符串的列表;例如“×”
,“theValue”
,或“A,B”
。functionBody
一个含有包括函数定义的
JavaScript
语句的字符串。接下来看两个例子:
你可能不知道在
ES3、ES5
中undefined
是能修改的可能大部分人不知道。
ES5
中虽然在全局作用域下不能修改,但在局部作用域中也是能修改的,不信可以复制以下测试代码在控制台执行下。虽然一般情况下是不会的去修改它。所以判断一个变量
a
是不是undefined
,更严谨的方案是typeof a === 'undefined'
或者a === void 0;
这里面用的是
void
,void
的作用是计算表达式,始终返回undefined
,也可以这样写void(0)
。更多可以查看
韩子迟
的这篇文章:为什么用「void 0」代替「undefined」解决了这几个问题,比较容易实现如下代码。
使用
new Function()
模拟实现的apply
利用模拟实现的
apply
模拟实现call
细心的你会发现注释了这一句
argsArray.push(arguments[i + 1]);
,事实上push
方法,内部也有一层循环。所以理论上不使用push
性能会更好些。面试官也可能根据这点来问时间复杂度和空间复杂度的问题。行文至此,就基本结束了,你可能还发现就是写的非严格模式下,
thisArg
原始值会包装成对象,添加函数并执行,再删除。而严格模式下还是原始值这个没有实现,而且万一这个对象是冻结对象呢,Object.freeze({})
,是无法在这个对象上添加属性的。所以这个方法只能算是非严格模式下的简版实现。最后来总结一下。总结
通过
MDN
认识call
和apply
,阅读ES5
规范,到模拟实现apply
,再实现call
。就是使用在对象上添加调用
apply
的函数执行,这时的调用函数的this
就指向了这个thisArg
,再返回结果。引出了ES6 Symbol
,ES6
的扩展符...
、eval
、new Function()
,严格模式等。事实上,现实业务场景不需要去模拟实现
call
和apply
,毕竟是ES3
就提供的方法。但面试官可以通过这个面试题考察候选人很多基础知识。如:call
、apply
的使用。ES6 Symbol
,ES6
的扩展符...
,eval
,new Function()
,严格模式,甚至时间复杂度和空间复杂度等。读者发现有不妥或可改善之处,欢迎指出。另外觉得写得不错,可以点个赞,也是对笔者的一种支持。
扩展阅读
《JavaScript设计模式与开发实践》- 第二章 第 2 章 this、call和apply
JS魔法堂:再次认识Function.prototype.call
不用call和apply方法模拟实现ES5的bind方法
JavaScript深入之call和apply的模拟实现
关于
作者:常以若川为名混迹于江湖。前端路上 | PPT爱好者 | 所知甚少,唯善学。
个人博客
segmentfault
前端视野专栏,开通了前端视野专栏,欢迎关注~掘金专栏,欢迎关注~
知乎前端视野专栏,开通了前端视野专栏,欢迎关注~
github blog,求个
star
^_^~The text was updated successfully, but these errors were encountered: