原生JS实现call和apply

call

先来看官方解释:

call()方法调用一个函数,其具有一个指定的this值和分别地提供的参数(参数的列表)。

也就是说使用call方法,你可以指定被调用函数的this值,并且给他传参数。

接下来我们自己实现一个call方法。

1、模拟如何指定this的值
先来看一段代码

function func() {
console.log(this.value)
}
var obj = {
value: 233
}
func.call(obj); //233
就是在调用func的时候,我们设置了当前this为obj,所以输出的this.value 就是233。不多说了。

试想一下,call方法是要改变this的值,我们可以给obj对象加一个属性fn,把func函数赋值给它,调用obj.fn,this自然就指向对象obj了,也起到了调用func函数的作用。所以我们可以这么实现:

  1. 给对象obj添加fn属性,并赋值func函数
    2、调用它–> obj.fn()
    3、delete obj.fn

function func() {
console.log(this.value)
}
var obj = {
value: 233
}
Function.prototype.mycall = function(obj) {

obj.fn = this; //这里的this就是调用mycall的func函数
obj.fn();
delete.fn;

}
func.mycall(obj); //233
ok,现在我们已经实现了如何指定的this的值。
2、实现可传参
Function.prototype.mycall = function(obj) {

var args = [];
for(var i = 1, len = arguments.length; i < len; i++) {
   args.push('arguments[' + i + ']');
}
obj.fn = this;
eval('obj.fn('+ args +')’);
delete obj.fn;

}
多说一句,我们拿到的args数组其实是[‘arguments[0]’, ‘arguments[1]’,…]这样子的,如果我们直接去拿arguments[0]的值的话,你测试一下传一个字符串参数,像这样,func.mycall(obj, ‘dd’);就会报错,告诉你dd is not defined。 因为eval会把字符串解析成一个变量。

这里遇到了一个问题,如何给一个函数传递不定参? 或者说怎么把不定长的数组转换成多个参数传递给函数??
三种办法: eval、apply和es6 的解构语法

1、eval(‘obj.fn(’+ args +’)’);
2、obj.fn.apply(obj, args);
3、obj.fn(…args);//es6解构语法
3、完善小细节
到这里,我们基本已经实现了call方法的功能,还有两个点需要注意:
– 如果第一个参数为null,this指向window
– 函数调用是有返回值的

呈上终极代码。

Function.prototype.mycall = function(obj) {

var args = [];
for(var i = 1, len = arguments.length; i < len; i++) {
   args.push('arguments[' + i + ']');
}
obj = obj || window;
obj.fn = this;
var result = eval('obj.fn('+ args +')’);
delete obj.fn;
return result;

}

apply

讲完了call方法,我们来说一下apply方法,apply方法和call方法非常类似。同样,先来看一下官方的解释:
apply()方法调用一个函数,其具有一个指定的this值,以及作为一个数组(或类数组对象)提供的参数。 注意:call()方法的作用和 apply() 方法类似,只有一个区别,就是 call()方法接受的是若干个参数的列表,而apply()方法接受的是一个包含多个参数的数组。

好,我们知道apply和call方法的区别就在于,call是传参数列表,apply是传一个数组。话不多说,开始写代码。

Function.prototype.myapply = function(obj, arr) {

obj = obj || window;
obj.func = this;
var result;
if(!arr) {
    result = obj.func();
} else {
    var args = [];
    for (var i = 0, len = arr.length; i < len; i++) {
        args.push('arr[' + i + ']');
    }
    result = eval('obj.func('+ args +')');
}
delete obj.func;
return result;

}

原文链接:https://blog.csdn.net/Smallsun_229/article/details/80721758

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
call、apply和bind都是用来改变函数中的this指向的方法。其中,call和apply可以直接调用函数并传递参数,而bind则返回一个新的函数,需要手动调用。 具体实现方案如下: - call的实现: 1. 给想要绑定的对象设置一个属性,并将该属性指向需要调用的函数。 2. 使用该对象调用函数,并传递参数。 3. 结束调用后,删除该属性。 - apply实现: 1. 给想要绑定的对象设置一个属性,并将该属性指向需要调用的函数。 2. 使用该对象调用函数,并传递参数数组。 3. 结束调用后,删除该属性。 - bind的实现: 1. 创建一个新的函数,并将原函数作为其中的属性保存起来。 2. 当新函数被调用时,将之前绑定的对象作为this,并传递参数。 3. 返回新函数供后续调用。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [如何实现call、apply、bind](https://blog.csdn.net/XIAO_A_fighting/article/details/116701887)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* [原生JS实现 call apply bind](https://download.csdn.net/download/weixin_38628990/14046564)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值