前端造轮子 手动实现call apply bind

这篇博客分享了作者在面试中的体会,强调前端开发者需要深入掌握基础知识。作者通过手动实现`call`、`apply`和`bind`这三个JavaScript函数,详细介绍了实现过程中的关键点和常见问题,如处理无参数、多个参数的情况,以及`apply`的扩展运算符应用和`bind`中`new`操作符的影响。
摘要由CSDN通过智能技术生成

最近面试的心得,前端还是得学精学细啊,所以我打算手动造几个轮子

参考博文 手写call、apply、bind实现及详解

知识点准备

  • call

  • apply

  • bind


手写一个call

// An highlighted block
var Person = {
	  name: 'Richard',
	  say() {
	  	console.log(this);
	  	console.log(`我叫${this.name}`)	
	   },
	  
}
var Person1 = {
	  name: 'Silcence'
}
Person.say.call(Person1); //我叫Silence
  • 初步实现

// An highlighted block
Function.prototype.Mycall = function(context){
 	console.log(this); //在Function类的原型上是否有say 
	//如果有, context.say 就是this 
    context.say = this;
	context.say();	
}
Person.say.Mycall(Person1)

在Function的原型上找到say方法,并赋予context

  • 问题所在

    • 未考虑无参数传递时的情况
      在这里插入图片描述
      未传参时 this指向Window

    • 未考虑多个参数的情况,若有多个参数应把参数传递给扩展方法

    • 给上下文定义的应是一个唯一的方法

  • 处理后的版本

    • Symbol()函数会返回symbol类型的值,每个从Symbol()返回的值都是唯一的
      console.log(Symbol(‘foo’) === Symbol(‘foo’));
    var Person = {
       name: 'Richard',
       say(name2) {
     	  	console.log(this);
     	  	console.log(`我叫${this.name}`)	
     	  	console.log(`这是传入的参数:${name2}`);
        },
       
     }
     var Person1 = {
     	  name: 'Silcence'
     }
     Function.prototype.Mycall = function(context){
      	console.log(this); //在Function类的原型上是否有say 
     	//如果有, context.say 就是this
     	
     	context = context||window // 参数为空时的处理 
         let fn = Symbol(context); //让context唯一
     	context.fn = this;
     	let arg = [...arguments].slice(1);  //除掉第一个参数,剩余的为传入参数   ...把类数组转换为真正的数组
     	context.fn(...arg)//执行fn
         delete context.fn //删除方法		
     }
      //Person.say.call(Person1);	
     
     //自定义call方法 
      Person.say.Mycall(Person1,'八哥');
    

    手写一个apply

    • ES6扩展运算符

      具体用法参照此篇文章 ES6扩展运算符

      和call方法大致相同,传参时第二个参数为数组

    var Person = {
      name: 'Richard',
      say(name2) {
      	console.log(this);
      	console.log(`我叫${this.name}`)
    	console.log(`这是传入的参数:${name2}`);	
       },
    	  
    }
    var Person1 = {
    	  name: 'Silcence'
    }
    Function.prototype.Mycall = function(context){
     	console.log(this); //在Function类的原型上是否有say 
    	//如果有, context.say 就是this
    	
    	context = context||window // 参数为空时的处理 
        let fn = Symbol(context); //让context唯一
    	context.fn = this;
    	let arg = [...arguments].slice(1);
    	context.fn(...arg)//执行fn
        delete context.fn //删除方法		
    }
     //Person.say.call(Person1);	
    
    //自定义call方法 
     Person.say.Mycall(Person1,['八哥']);
    

手写一个Bind

  • Bind的定义

    fun.bind(thisArg[, arg1[, arg2[, …]]])

    • bind() 函数会创建一个新的绑定函数
    • 注意,一个 绑定函数 也能使用 new 操作符创建对象,这种行为就像把原函数当成构造器,thisArg 参数无效。也就是 new 操作符修改 this 指向的优先级更高。
  • 初步实现

var Person = {
     name: 'Richard',
     say(name2) {
     	console.log(this);
     	console.log(`我叫${this.name}`)
   	console.log(`这是传入的参数:${name2}`);	
      },
     
}
var Person1 = {
     name: 'Silcence'
}
Function.prototype.myBind = function(thisArg){
   if (typeof this!== 'function'){
   	return;
   }
   var that = this;
   var args = [...arguments].slice(1)
   return function(){
   	//因为同样支持柯里化形式的传参,我们需要再次存储参数
   	let newArg = [...arguments]
   	console.log(newArg)
   	return that.apply(thisArg,args.concat(newArg));	
   }	
}

var result = Person.say.myBind(Person1)
result('八哥');	

关于柯里化传参

  • List item
  • new操作符的坑

    • 正如上文所提,来new一个绑定函数
    • //todo 这里我还不太理解
Function.prototype.bind2 = function(context) {
 if (typeof this !== "function") {
   throw new TypeError("Error");
 }

 const that = this;
 // 保留之前的参数,为了下面的参数拼接
 const args = [...arguments].slice(1);

 return function F() {
   // 如果被new创建实例,不会被改变上下文!
   if (this instanceof F) {
     return new that(...args, ...arguments);
   }

   // args.concat(...arguments): 拼接之前和现在的参数
   // 注意:arguments是个类Array的Object, 用解构运算符..., 直接拿值拼接
   return that.apply(context, args.concat(...arguments));
 };
};

/**
* 以下是测试代码
*/

function test(arg1, arg2) {
 console.log(arg1, arg2);
 console.log(this.a, this.b);
}

const test2 = test.bind2(
 {
   a: "a",
   b: "b"
 },
 1
); // 参数 1

test2(2); // 参数 2
call、applybind都是用来改变函数中的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、applybind](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、付费专栏及课程。

余额充值