首頁 > 軟體

JS面試中你不知道的call apply bind方法及使用場景詳解

2023-02-08 22:01:28

面試

面試官 :說一下 call apply bind 的三者區別吧?

我:啊.....這。

三者區別

call

我們先來看看 call call 函數接收兩個引數

@params1 需要指向this

@params2 需要攜帶的引數

就拿這段程式碼 來說 我呼叫 foo 函數 的時候 想去 執行obj 中的 eat函數 怎麼辦?

預設 我的 foo 裡面什麼頁面有 怎麼可能會列印 吃飯呢?this預設指向 window ,window 物件中 沒有 eat 方法不應該 報錯 not defind 嗎?

let obj = {
  eat: function (args) {
    console.log("吃飯", args);
  }
}
function foo(...args) {
  this.eat(args)
}

你可以這樣 , 此時呼叫的時候 foo 的函數 this 就指向了 obj 但是隻是這一次呼叫哦 ,下次 this 是指向 window

console.log(foo.call(obj, 123, 456)); / / 吃飯 [ 123, 456 ]

呼叫結果 : 傳入 this 和 引數 立即執行函數

apply.

@params1 需要指向this

@params2 需要攜帶的引數 傳入的引數資料型別為 Array

let obj = {
  eat: function (args) {
    console.log("吃飯", args);
  }
}
function foo(...args) {
  this.eat(args)
}
console.log(foo.apply(obj, [1, 2, 3])); // 吃飯 [1,2,3 ]

呼叫結果 : 傳入 this 和 引數,立即執行

bind

@params1 需要指向this

@params2 需要攜帶的引數

let obj = {
  eat: function (args) {
    console.log("吃飯", args);
  }
}
function foo(...args) {
  this.eat(args)
}
console.log(foo.bind(obj, 1, 2, 3)()); //吃飯 [ 1, 2, 3 ]

呼叫結果 : 傳入 this 和 引數 返回一個 新的函數,不會立即執行

總結

call 函數 傳入繫結 this的物件 ,攜第二個引數以 參數列形式傳入 並會立即執行

apply 函數 傳入繫結 this的物件 第二個引數以 陣列的形式 傳入 並會立即執行

bind 函數 傳入繫結 this的物件 第二個引數以 陣列或參數列的形式傳入 不會立即執行函數會返回一個新的函數 自由呼叫

應用場景

利用call 實現 建構函式原型繼承

function Person(name, age, friends) {
  this.friends = friends
  this.name = name
  this.age = age
}
Person.prototype.eat = function () {
  console.log(`${this.name}在吃飯`);
}
Person.prototype.running = function () {
  console.log(`${this.name}去跑步了`);
}
function Student(study, name, age, friedns) {
  Person.call(this, name, age, friedns)
  this.study = study
}
function Teacher(plaseLoveStudent) {
  this.plaseLoveStudent = plaseLoveStudent
}
//繼承 person 
Student.prototype = new Person()
const stu = new Student('語文', "張三", 18, ['王安石'])
const stu2 = new Student('數學', "李四", 18, ['哈利波特'])
const tec = new Teacher("王安怡")
console.log(stu === stu2);
console.log(stu);
console.log(stu.friends);
console.log(stu2.friends);

簡單實現

call

實現思路 :

1:在Function 原型身上新增一個方法。

2:mycall 接受 兩個引數 一個是繫結的 this,還有就是 參數列

3:儲存呼叫者,其實這裡更嚴謹一點還需要判斷呼叫者的 型別

4:判斷傳入的thisArgs 是不是 undefined 或者 null 如果是 則 this指向 window 否則將 繫結的 this封裝成 一個物件

5:然後將 函數本身儲存在 上面封裝好的物件中

6:呼叫並傳入args

7:完成實現 this指向的改變

Function.prototype.mycall = function (thisArgs, ...args) {
  /**
   * 
   * 儲存this(當前呼叫者)
   * 
   */
  const newFunc = this
  // 在當前呼叫者 this身上 儲存 呼叫者
  thisArgs = (thisArgs !== undefined && thisArgs !== null) ? Object(thisArgs) : window
  thisArgs.func = newFunc
  thisArgs.func(...args)
}

apply

apply 就不說了 實現思路是一樣的 只不過傳入的引數不一樣做一個判斷就行了

Function.prototype.myapply = function (thisArgs, argArray) {
  if (!(argArray instanceof Array)) {
    throw '引數型別限定為 Array'
  }
  else {
    const func = this
    thisArgs = (thisArgs !== undefined && thisArgs !== null) ? Object(thisArgs) : window
    thisArgs.fn = func
    thisArgs.fn(argArray)
  }
}

bind

bind 跟 apply call 有一些區別

bind 會返回一個新的函數 , 所以我們在內部需要自己定義一個 函數 給返回出去, 並且,可能出現 返回新函數呼叫時 繼續傳入引數 所以我們需要將引數合併

Function.prototype.mybind = function (thisArgs, ...argArray) {
  const oldFunc = this
  console.log(this);
  thisArgs = (thisArgs !== undefined && thisArgs !== null) ? Object(thisArgs) : window
  function proxyFn(...args) {
    thisArgs.func = oldFunc
    const funcCallResult = thisArgs.func([...argArray, ...args]) // 合併兩次的 引數
    delete thisArgs.func
    return funcCallResult
  }
  return proxyFn
}

謝謝,到此就完成了簡單版的 call apply bind 的實現,更多關於JS call apply bind方法的資料請關注it145.com其它相關文章!


IT145.com E-mail:sddin#qq.com