likes
comments
collection
share

你会写显示绑定方法吗?

作者站长头像
站长
· 阅读数 41

1. 前言

2. call() 方法

定义: 调用一个对象的方法,以另一个对象换当前对象,可将一个函数的对象上下文从初始的上下文改变为由thisObj 指定的新对象。

语法: Function.call(thisObj, arg1, arg2, ....)

说明:

  • 如果没提供thisObj参数,则会指向全局Window
  • call接受多个参数为Function的参数,调用后返回Funtion函数的执行结果。

实现call关键因素:

  • 使Funtion 函数的this指向thisObj。
  • 将Funtion 函数执行,同时将后序参数作为funtion的参数,若有返回值则返回。
Function.prototype.my_call = function() {
  if (typeof this !== 'function') { // 优化函数才可以调用`call`
    throw new TypeError('error')
  }
  let obj = arguments[0]
  const args = Array.prototype.slice.call(arguments, 1) // 第一个函数以外的参数
  obj = obj || window  // 防止没传第一个参数
  const fn = Symbol('fn') // 声明 唯一变量 fn
  obj[fn] = this // 改变函数this指向(关键代码)
  const result = obj[fn](...args)  // 调用函数(关键代码)
  delete obj[fn] // 删除 obj 上的 fn 属性
  return result  // 返回函数执行结果
}
var name = 'zhangsan'
var obj = {
  name: 'lisi',
  fn: 123
}
//我们实现
const res = foo.my_call(obj, 1, 2) 
console.log(res); //  lisi 3 
 // 官方
const res1 = foo.call(obj, 1, 2) //  lisi 3 
console.log(res1); //  lisi 3 

3. apply() 方法

定义: 与call方法相同,但apply第二个参数接受一个数组。

语法: Function.call(thisObj, [arg1, arg2, ...])

说明:

  • 如果没提供thisObj参数,则会指向全局Window
  • apply第二个参数为数组,调用后返回Funtion函数的执行结果。

实现apply关键因素:

  • 使Funtion 函数的this指向thisObj。
  • 将Funtion 函数执行,同时将第二个参数作为Funtion的参数,返回Function的执行结果。
Function.prototype.my_apply = function(obj, args) {
  if (typeof this !== 'function') { // 优化函数才可以调用`apply`
    throw new TypeError('error')
  }
  obj = obj || window  // 防止没传第一个参数
  const fn = Symbol('fn') // 声明 唯一变量 fn
  obj[fn] = this  // 改变函数this指向(关键代码)
  const res = obj[fn](...args)  // 调用函数(关键代码)
  delete obj[fn] // 删除 obj 上的 fn 属性
  return res // 返回函数执行结果
}
var name = 'zhangsan'
var obj = {
  name: 'lisi',
  fn: 123
}
//我们实现
const res = foo.my_apply(obj, [1, 2]) 
console.log(res); //  lisi 3 
 // 官方
const res1 = foo.aplly(obj, [1, 2]) //  lisi 3 
console.log(res1)  //  lisi 3 

4. bind() 方法

定义: 与call方法相同, 可接受任意参数。

语法: Function.call(thisObj, arg1, arg2, ...)

说明:

  • 如果没提供thisObj参数,则会指向全局Window
  • bind接受任意个参数,调用后返回一个新的绑定函数。

实现bind关键因素:

  • 使Funtion 函数的this指向thisObj。
  • 将Funtion 函数执行,同时其他参数作为Funtion的参数, 返回一个新的绑定函数。
  • 如果 new 了返回的新的绑定的函数,该函数的this需要指向Function函数而不是thisObj。
Function.prototype.my_bind = function(obj) {
  if (typeof this !== 'function') { // 优化函数才可以调用`bind`
    throw new TypeError('error')
  }
  obj = obj || window  // 防止没传第一个参数
  const args = Array.prototype.slice.call(arguments, 1)  // 第一个函数以外的参数
  const _this = this 
  const pro = function() {}
  if (this.prototype) {
    pro.prototype = this.prototype
  }
  const bound = function () { // 声明一个bound函数
    // 判断该函数是否被new
    return _this.apply(  // 掉用 apply 方法改变 this指向
      this instanceof pro ? this : obj, // 判断 bound 函数时候被 new 了  如果 new this 指向foo 反之,指向 obj
      args.concat(Array.prototype.slice.call(arguments))
    )
  }
  bound.prototype = new pro() //  bound 函数的原型继承到了foo的原型  (原型链继承)
  return bound // 返回 bound 函数
}
var name = 'zhangsan'
var obj = {
  name: 'lisi',
  fn: 123
}
function foo(a, b) {
  console.log(this.name, a + b);
  return a + b
}
// 自己实现
const bar = foo.my_bind(obj)
console.log(bar());
 let bar1 = new bar(3, 4)
 console.log(bar1);
// 官方
const bar2 = foo.bind(obj)
console.log(bar2());
let bar3 = new bar(3, 4)
console.log(bar3);

你会写显示绑定方法吗?

5. 结语

你会写显示绑定方法吗? 在不同的开发环境下,我们灵活的运用这三种显示绑定方法,可以使我们的this永远不会丢失。

转载自:https://juejin.cn/post/7175534540381749306
评论
请登录