了解Javascript中函數(shù)作為對(duì)象的魅力
前言
Javascript賦予了函數(shù)非常多的特性,其中最重要的特性之一就是將函數(shù)作為第一型的對(duì)象。那就意味著在javascript中函數(shù)可以有屬性,可以有方法, 可以享有所有對(duì)象所擁有的特性。并且最重要的,她還可以直接被調(diào)用
我們簡單的試驗(yàn)一下就可以發(fā)現(xiàn)
// 簡單實(shí)驗(yàn) 函數(shù)作為對(duì)象的存在 let fn = function () {} fn.prop = 'fnProp' console.log(fn.prop) // fnProp
為函數(shù)添加屬性的這個(gè)特性我覺的大家在平時(shí)的開發(fā)中基本沒什么嘗試或者是使用過,但是在一些JS庫或者是事件回掉管理中都能發(fā)揮出很大的用處。下面一起來看幾個(gè)例子。
函數(shù)緩存
在某有一些的情況下我們可以要存儲(chǔ)一組相關(guān)但是相互又獨(dú)立的函數(shù)。這個(gè)需求看起來很easy,實(shí)現(xiàn)起來也不復(fù)雜。最顯而易見的做法是使用一個(gè)數(shù)組來保存所有的函數(shù),
這樣不是不可以,但是顯然這種做法不是最好的。下面通過為函數(shù)屬性我們呢來實(shí)現(xiàn)這個(gè)我們的目的
// 1:函數(shù)緩存示例 let store = { nextId: 1, // id cache: {}, // 緩存 add (fn) { // 如果函數(shù)中沒有id屬性那么就緩存 if (!fn.id) { console.log(`begin add func ${fn.name}`) fn.id = store.nextId ++ // 設(shè)置完緩存之后返回true return !!(store.cache[fn.id] = fn) } else { console.log(`${fn.name} is already in cache`) } } } function storeCache() {} store.add(storeCache) // begin add func storeCache store.add(storeCache) // storeCache is already in cache
上面的這一段代碼邏輯清晰,store對(duì)象用來管理我們的緩存,cache屬性用來存儲(chǔ)函數(shù),nextId屬性用來保存當(dāng)前的緩存Id,add()方法用來設(shè)置存儲(chǔ),先來判斷當(dāng)前函數(shù)是否已經(jīng)在緩存中然后再去設(shè)置緩存,這樣就能限制函數(shù)的重復(fù)添加,最后返回true。
!!構(gòu)造是一種可以將任意Javascript表達(dá)式轉(zhuǎn)化為其等效布爾值的簡單方式。
緩存記憶函數(shù)
這種函數(shù)可以記住之前已經(jīng)計(jì)算過的結(jié)果,避免了不必要的計(jì)算,這顯然是能夠提升代碼性能的。
在舉例之前我們先來看看這種方式的優(yōu)缺點(diǎn)
優(yōu)點(diǎn)
- 緩存了之前的結(jié)果,最終用戶享有性能優(yōu)勢
- 實(shí)際上是發(fā)生在幕后,操作無感
缺點(diǎn)
- 內(nèi)存的犧牲這是肯定的
- 打破了存粹性(一個(gè)函數(shù)或者方法應(yīng)該只做好一件事)
- 如果方法中有算法,那么很難測量這個(gè)算法的性能
了解了優(yōu)缺點(diǎn)我們來看一個(gè)簡單的計(jì)算素?cái)?shù)的例子(不是很嚴(yán)謹(jǐn))
// 2: 緩存記憶函數(shù) function isPrime (value) { if (!isPrime.anwers) isPrime.anwers = {} // 先從緩存里面取 if (isPrime.anwers[value] != null ) { return isPrime.anwers[value] } // 開始進(jìn)行判斷和計(jì)算 let prime = value != 1 for (let index = 2; index < value; index++) { if (value % index == 0) { prime = false break; } } // 保存計(jì)算出來的值 return isPrime.anwers[value] = prime } console.log(isPrime(5)) console.log(`從函數(shù)記憶中直接讀取${isPrime.anwers[5]}`)
這里呢 好處是特別明顯的我們再次的取用isPrime.anwers[5]的時(shí)候不需要經(jīng)過任何的計(jì)算,但是大型的計(jì)算要主要內(nèi)存的使用
緩存記憶DOM元素
通過元素的標(biāo)簽查詢DOM的操作的的代價(jià)是昂貴的,各位前端大佬肯定都很清楚。我們下面使用緩存記憶的方式來進(jìn)行這個(gè)操作
// 3:緩存記憶DOM元素 function getElements (name) { if (!getElements.cache) getElements.cache = {} return getElements.cache[name] = getElements.cache[name] || document.getElementsByTagName(name); } console.log(getElements('div')) // HTMLCollection console.log(getElements.cache['div']) // HTMLCollection
這個(gè)函數(shù)和上面的緩存使用的同一個(gè)手法,而且這簡單的4句代碼能為我們的性能帶來大幅度的提升。這也算是一種超能力吧。函數(shù)的很多特性都和其上下文有關(guān),接下來我們研究一個(gè)和上下文又換的例子。
偽造數(shù)組方法(上下文相關(guān))
在一些情況下我們想創(chuàng)建一個(gè)包含一組數(shù)據(jù)的對(duì)象,但是這個(gè)數(shù)據(jù)包含很多的狀態(tài),比如和集合項(xiàng)有關(guān)的元數(shù)據(jù)那么我們用數(shù)組來存就不太合適了。那么這里我們就用對(duì)象的方式來假扮數(shù)組。通過改變上下文來完成一些“不法的行為”
// 4:偽造數(shù)組方法 // <input type="button" id="add" > // <input type="button" id="remove" > let elems = { length: 0, //為了保存?zhèn)€數(shù) add (elem) { Array.prototype.push.call(this, elem) }, gather (id) { this.add(document.getElementById(id)) } } elems.gather('add') elems.gather('remove') console.log(elems[0]); // <input type="button" id="add" > console.log(elems[1]); // <input type="button" id="remove" > console.log(elems.length); // 2 console.log(elems); /** 0: input#add 1: input#remove add: ƒ add(elem) gather: ƒ gather(id) length: 2 */
在我還對(duì)JS懵懵懂懂的時(shí)候看到這樣的操作被秀了一臉,簡直是刺激了我幼小的心靈。
我們在add函數(shù)中實(shí)現(xiàn)了把元素添加到了集合中,而且Array又正好提供push方法, 不用白不用。這種操作也是直白的展示了函數(shù)上下文的超強(qiáng)特性。
總結(jié)
Javascript強(qiáng)大的靈活性, 也帶來更多的可能性。 路漫漫其修遠(yuǎn)兮,吾將上下而求索。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
微信小程序?qū)崿F(xiàn)傳參數(shù)的幾種方法示例
這篇文章主要給大家介紹了關(guān)于微信小程序?qū)崿F(xiàn)傳參數(shù)的幾種方法,分別是navigator跳轉(zhuǎn)時(shí)、全局變量、列表index下標(biāo)取值以及form表單傳值的相關(guān)內(nèi)容,需要的朋友可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2018-01-01javascript元素動(dòng)態(tài)創(chuàng)建實(shí)現(xiàn)方法
這篇文章主要介紹了javascript元素動(dòng)態(tài)創(chuàng)建實(shí)現(xiàn)方法,涉及javascript操作元素的相關(guān)技巧,需要的朋友可以參考下2015-05-05js cavans實(shí)現(xiàn)靜態(tài)滾動(dòng)彈幕
這篇文章主要為大家詳細(xì)介紹了js cavans實(shí)現(xiàn)靜態(tài)滾動(dòng)彈幕,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-05-05javascript實(shí)時(shí)顯示北京時(shí)間的方法
這篇文章主要介紹了javascript實(shí)時(shí)顯示北京時(shí)間的方法,涉及javascript操作時(shí)間顯示的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03學(xué)習(xí)JavaScript設(shè)計(jì)模式(繼承)
這篇文章主要帶領(lǐng)大家學(xué)習(xí)JavaScript設(shè)計(jì)模式,其中重點(diǎn)介紹繼承,舉例說明為什么需要繼承,對(duì)繼承進(jìn)行詳細(xì)剖析,感興趣的小伙伴們可以參考一下2015-11-11一文帶你理解微信小程序中RPC通信的實(shí)現(xiàn)
在微信小程序開發(fā)中,要實(shí)現(xiàn)兩個(gè)線程之間的通信是一項(xiàng)重要的任務(wù),所以本文就來講講如何使用小程序的?postMessage?和?addListener?API?來實(shí)現(xiàn)在兩個(gè)線程之間進(jìn)行高效的?RPC?通信吧2023-06-06