call
简单的介绍call的作用:
我们可通过call()方法在指定this指向和传递若干个参数的条件下调用某个函数(function)。
举个简单的例子:
js
var obj = {
val: '2021'
}
function testCall(){
console.log(this.val);
}
testCall.call(obj);
// 2021
两点:
1.testCall函数中的this指向了obj
2.testCall函数执行了
模拟实现(第一步)
那么我们该怎么模拟实现这两个效果呢?
试想当调用 call 的时候,把 obj 对象改造成如下:
var obj = {
val: '2021',
testCall: function() {
console.log(this.val)
}
};
obj.testCall(); // 2021
这个时候 this 就指向了 obj,是不是很简单呢?
但是这样却给 obj 对象本身添加了一个属性,这可不行呐!
不过也不用担心,我们用 delete 再删除它不就好了~
所以我们模拟的步骤可以分为:
- 将函数设为对象的属性
- 执行该函数
- 删除该函数
js
Function.prototype.mycall = function(context){
context.fn = this;
context.fn();
delete context.fn;
}
测试一下:
js
var obj = {
val: '2021'
}
function testCall(){
console.log(this.val);
}
testCall.mycall(obj);
// PS C:\GitHub\Blog\demos\call&&apply模拟实现> node .\0301.js
// 2021
// PS C:\GitHub\Blog\demos\call&&apply模拟实现>
测试通过了。
模拟实现(第二步)
call还支持传递若干个函数参数来执行函数,而且传入的参数不定长。
比如:
js
function testCall1(name, age){
console.log(this.val, 'name ' + name, 'age ' + age);
}
testCall1.call(obj, '小明', '18');
// 2021 name 小明 age 18
处理参数,然后有了第二版的代码:
js
Function.prototype.mycall = function(context){
context.fn = this;
var args = [];
for(var i = 1; i < arguments.length; i++){
args.push('arguments[' + i + ']');
}
eval('context.fn(' + args +')');
delete context.fn;
}
模拟实现(第三步)
注意下两个小细节:
- this可以传入null,并且此时this指向window
- 函数是有返回值的
第三版代码:
JS
Function.prototype.mycall = function(context, firstParam){
var context = Object(context) || window;
var result;
context.fn = this;
if (!firstParam) {
result = context.fn();
} else {
var args = [];
for(var i = 1; i < arguments.length; i++){
args.push('arguments[' + i + ']');
}
result = eval('context.fn(' + args +')');
}
delete context.fn;
return result;
}
js
// 用Es6实现
Function.prototype.mycall = function(context, ...argu) {
context = context || window
const funcid = Symbol()
context[funcid] = this // 调用call的函数
const result = context[funcid](...argu)
delete context[funcid]
return result
}
测试一下:
js
var obj = {
val: '2021'
}
function testCall(name, age){
console.log(this.val, 'name' + name, 'age' + age);
return this.val;
}
console.log(testCall.mycall(obj, '小明', '18'));
// 2021 name小明 age18
// 2021
apply
apply和call的作用类似,只不过传递参数时,apply要将所有参数放到一个数组中。
举个例子:
js
var obj = {
val: '2021'
}
function testCall(name, age){
console.log(this.val, 'name' + name, 'age' + age);
return this.val;
}
console.log(testCall.apply(obj, ['小明', '18']));
实现几乎一致,直接粘代码了
js
Function.prototype.myApply = function(context, paramArr){
var context = Object(context) || window;
var result;
context.fn = this;
if (!paramArr || !Array.isArray(paramArr)) {
result = context.fn();
} else {
var args = [];
for(var i = 0; i < paramArr.length; i++){
args.push('paramArr[' + i + ']');
}
result = eval('context.fn(' + args +')');
}
delete context.fn;
return result;
}
js
// ES6
Function.prototype.myApply = function(ctx, argu = []){
ctx = ctx || window
const funcId = Symbol()
ctx[funcId] = this
const result = ctx[funcId](...argu)
Reflect.deleteProperty(ctx, 'funcId')
return result
}