JS前端面試手寫apply和bind實(shí)例
前言
面試官問:“聊一聊你理解的 apply
和 bind
。”
于是我便開始開始介紹起這兩個(gè)知識點(diǎn),最后順帶提了它們的實(shí)現(xiàn)代碼。
這不提倒還好,一提就出了大事,一下子給面試官找到了面試題目。
面試官緊接著說:“既然你提到了代碼,那就手寫一下它倆吧。”
我一下子不知所措。雖然我了解過 apply
和 bind
手寫代碼,但是現(xiàn)在讓我立馬手寫出來它們,實(shí)屬有些困難。
不過最后還是硬著頭皮寫了一下,但是結(jié)果不盡人意。
apply && bind
函數(shù) apply
和 bind
在日常開發(fā)中經(jīng)常會(huì)被用到,理解它們的作用以及邏輯至關(guān)重要。所以今天我們來探討一下它們的實(shí)現(xiàn)邏輯,并對它們進(jìn)行手寫,來學(xué)會(huì)如何輕松手寫它們。
apply && bind 作用
知己知彼,方能百戰(zhàn)百勝。在進(jìn)行手寫之前,我們先來簡單聊一下它們。 眾所周知,apply 和 bind 作用是改變函數(shù)的調(diào)用對象。很多人對這個(gè)不是很理解,什么是函數(shù)的調(diào)用對象呢?其實(shí)簡單點(diǎn)理解就是改變函數(shù)的 this
對象指向。因?yàn)檫@時(shí)候 this
在函數(shù)就充當(dāng)了一個(gè)調(diào)用的作用,而 apply 和 bind 就是有著改變 this
指向的作用。
apply
的其他作用,是改變對象的執(zhí)行上下文,并且是立即執(zhí)行的。 bind
也能改變對象的執(zhí)行上下文。
相同點(diǎn) VS 不同點(diǎn)
它們的相同點(diǎn)在于都可以改變 this
的指向,并且傳入的第一個(gè)參數(shù)都是綁定的 this
,不同點(diǎn)在于 bind 返回的是一個(gè)改變了 this
指向的函數(shù),便于稍后調(diào)用,而 apply 會(huì)立即調(diào)用。另外 apply 是一次性傳入?yún)?shù),而 bind 可以分為多次傳入。
輕松手寫
有了理論基礎(chǔ),我們可以開始手寫部分了。
手寫實(shí)現(xiàn) apply
我們先來看 apply,apply 手寫實(shí)現(xiàn)其實(shí)很簡單,大致思路如下:
首先用 typeof
來檢查調(diào)用 apply
的對象是否為函數(shù),如果不是則拋出錯(cuò)誤。然后將函數(shù)作為傳入的 context
對象的一個(gè)屬性,并調(diào)用該函數(shù)。 最后調(diào)用之后通過 delete
刪除該屬性,避免對傳入對象造成污染。context
代表上下文對象。
代碼如下:
Function.prototype.apply = function (context, args) { if (typeof this !== 'function') { console.log('not a function') } const fn = Symbol() context[fn] = this context[fn](...args) delete context[fn] }
這里使用到了新的 Symbol
數(shù)據(jù)類型,主要是避免在把函數(shù)賦值給 context
對象的時(shí)候,因?yàn)閷傩悦麤_突而覆蓋掉原有屬性。
手寫實(shí)現(xiàn) bind
相比較與 apply 手寫,我個(gè)人覺得 bind 手寫會(huì)相對復(fù)雜一些。bind 手寫實(shí)現(xiàn)思路如下:
和之前一樣,首先判斷調(diào)用對象是否為函數(shù),然后獲取其余傳入?yún)?shù)值,并創(chuàng)建一個(gè)函數(shù)返回。最后根據(jù)調(diào)用方式,傳入不同的綁定值。
函數(shù)內(nèi)部使用 apply 來綁定函數(shù)調(diào)用,需要判斷函數(shù)作為構(gòu)造函數(shù)的情況,這個(gè)時(shí)候需要傳入當(dāng)前函數(shù)的 this
給 apply 調(diào)用,其余情況都傳入指定的上下文對象 context
。
代碼如下:
Function.prototype.myBind = function(context) { if (typeof this !== "function") { console.log("Error"); } let args = [...arguments].slice(1), fn = this; return function Fn() { return fn.apply( this instanceof Fn ? this : context, args.concat(...arguments) ); }; };
總結(jié)
關(guān)于它們的手寫就到這里了,相信下次再面對同樣的問題時(shí)處理起來就能游刃有余了。這兩個(gè)手寫各有各的特點(diǎn),我個(gè)人建議可以多寫寫 bind 的手寫,我覺得它會(huì)比 apply 出錯(cuò)率更高。
雖然文章中是以手寫 apply 和 bind 為主,但是并不缺少 js
相關(guān)知識點(diǎn),比如 this
指向問題,總之,js
很重要,既是基礎(chǔ)又是重點(diǎn)。
以上就是手寫apply和bind實(shí)例的輕松實(shí)現(xiàn)詳解的詳細(xì)內(nèi)容,更多關(guān)于apply bind手寫實(shí)例的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
一文了解JavaScript用Element?Traversal新屬性遍歷子元素
這篇文章主要介紹了一文了解JavaScript用Element?Traversal新屬性遍歷子元素,文章圍繞Element?Traversal新屬性的相關(guān)資料展開詳細(xì)內(nèi)容,需要的朋友可以參考一下,希望對大家有所幫助2021-11-11umi插件開發(fā)仿dumi項(xiàng)目實(shí)現(xiàn)基礎(chǔ)路由解析
這篇文章主要為大家介紹了umi插件開發(fā)仿dumi項(xiàng)目實(shí)現(xiàn)基礎(chǔ)路由解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01微信小程序 tabs選項(xiàng)卡效果的實(shí)現(xiàn)
這篇文章主要介紹了微信小程序 tabs選項(xiàng)卡效果的實(shí)現(xiàn)的相關(guān)資料,微信小程序內(nèi)部組件沒有Tabs 選項(xiàng)卡的功能,自己實(shí)現(xiàn)個(gè)類似的,需要的朋友可以參考下2017-01-01微信小程序?qū)崿F(xiàn)拖拽 image 觸摸事件監(jiān)聽的實(shí)例
這篇文章主要介紹了微信小程序?qū)崿F(xiàn)拖拽 image 觸摸事件監(jiān)聽的實(shí)例的相關(guān)資料,這里提供image觸摸并監(jiān)聽的簡單實(shí)例,需要的朋友可以參考下2017-08-08前端JavaScript徹底弄懂函數(shù)柯里化curry
隨著主流JavaScript中函數(shù)式編程的迅速發(fā)展, 函數(shù)柯里化在許多應(yīng)用程序中已經(jīng)變得很普遍。 了解它們是什么,它們?nèi)绾喂ぷ饕约叭绾纬浞掷盟鼈兎浅V匾?。本篇文章小編九向大家詳?xì)介紹JavaScript函數(shù)柯里化,需要的小伙伴可以參考下面文字內(nèi)容2021-09-09JS版的date函數(shù)(和PHP的date函數(shù)一樣)
這篇文章主要介紹了JS版的date函數(shù),使用方法和PHP的date函數(shù)一樣,需要的朋友可以參考下2014-05-05