Javascript 高性能之遞歸,迭代,查表法詳解及實(shí)例
Javascript 高性能之遞歸,迭代,查表法詳解
遞歸
概念:函數(shù)通過(guò)直接調(diào)用自身,或者兩個(gè)函數(shù)之間的互相調(diào)用,來(lái)達(dá)到一定的目的,比如排序,階乘等
簡(jiǎn)單的遞歸
階乘
function factorial(n) {
if (n == 0) {
return 1;
} else {
return n * factorial(n - 1);
}
}
遞歸實(shí)現(xiàn)排序
/*
排序且合并數(shù)組
*/
function myMerge(left, right) {
// 保存最后結(jié)果的數(shù)組
var res = [];
// 有一個(gè)數(shù)組結(jié)束了就結(jié)束排序,一般情況下,兩個(gè)數(shù)組長(zhǎng)度應(yīng)該保持一樣
while (left.length > 0 && right.length > 0) {
if ( left[0] < right[0] ) {
// 把left第一個(gè)成員刪除,存儲(chǔ)到新數(shù)組res中
res.push(left.shift());
} else {
res.push(right.shift());
}
}
// 如果還有剩余直接合并到新數(shù)組
return res.concat(left).concat(right);
}
/*
遞歸方式
*/
function recursion(items) {
if (items.length == 1) {
return items;
}
var middle = Math.floor(items.length / 2),
left = items.slice(0, middle), // 取數(shù)組前半部分
right = items.slice(middle); // 取數(shù)組后半部分
// 遞歸排序
return myMerge(recursion(left), recursion(right));
}
迭代
每個(gè)瀏覽器對(duì)遞歸都有調(diào)用棧上限的問(wèn)題,且如果不小心使用也很容易造成死循環(huán)導(dǎo)致程序崩潰。如果考慮到性能問(wèn)題,可以使用迭代來(lái)代替遞歸,因?yàn)檫\(yùn)行循環(huán)總比反復(fù)調(diào)用函數(shù)的開(kāi)銷(xiāo)少很多
/*
迭代方式,不使用遞歸,可以避免出現(xiàn)棧溢出問(wèn)題
*/
function iteration(items) {
if (items.length == 1) {
return items;
}
var work = [];
// 將items成員全部轉(zhuǎn)化成數(shù)組,保存到數(shù)組中
for (var i = 0, len = items.length; i < len; i++) {
work.push([items[i]]);
}
work.push([]); // 數(shù)組長(zhǎng)度為奇數(shù)時(shí)
// 迭代
for (var lim = len; lim > 1; lim = (lim + 1) / 2) {
for (var j = 0, k = 0; k < lim; j++, k+=2) {
work[j] = myMerge(work[k], work[k + 1]);
}
work[j] = []; // 數(shù)組長(zhǎng)度為奇數(shù)時(shí)
}
return work[0];
}
/* 迭代過(guò)程分析
假設(shè): var test = [1, 3, 9, 7, 4, 8, 6, 5, 0, 2]; // len == 10
work = [[1], [3], [9], [7], [4], [8], [6], [5], [0], [2], []]; // len == 11;
// 迭代(二分法)
a) lim: 11 > 1
1) k = 0, work[0] = myMerge([1], [3]) ==> work[0] = [1, 3]
2) k = 2, work[1] = myMerge([9], [7]) ==> work[1] = [7, 9]
3) k = 4, work[2] = myMerge([4], [8]) ==> work[2] = [4, 8]
4) k = 6, work[3] = myMerge([6], [5]) ==> work[3] = [5, 6]
5) k = 8, work[4] = myMerge([0], [2]) ==> work[4] = [0, 2]
> 在后面添加個(gè)空數(shù)組是為了數(shù)組長(zhǎng)度為奇數(shù)時(shí)的情況,能有個(gè)對(duì)象做比較,否則會(huì)出現(xiàn)越界錯(cuò)誤
b) lim: 6 > 1, work = [[1,3], [7,9], [4,8], [5,6], [0,2], []];
1) k = 0, work[0] = myMerge([1,3], [7,9]) ==> work[0] = [1, 3, 7, 9]
2) k = 2, work[1] = myMerge([4,8], [5,6]) ==> work[1] = [4, 5, 6, 8]
3) k = 4, work[2] = myMerge([0,2], []) ==> work[2] = [0,2]
> 最后一個(gè)[]會(huì)被myMerge函數(shù)給合并,所以不用擔(dān)心添加的空數(shù)組對(duì)后續(xù)產(chǎn)生影響
c) lim: 3 > 1, work = [[1, 3, 7, 9], [4, 5, 6, 8], [0, 2], []];
1) k = 0, work[0] = myMerge([1, 3, 7, 9], [4, 5, 6, 8]) ==> work[0] = [1,3,4,5,6,7,8,9]
2) k = 2, work[1] = myMerge([0, 2], []) ==> work[1] = [0, 2]
d) lim: 2, work = [[1,3,4,5,6,7,8,9], [0,2], []];
1) k = 0, work[0] = myMerge([1,3,4,5,6,7,8,9], [0,2]) ==> work[0] = [0,1,2,3,4,5,6,7,8,9]
> 至此排序整個(gè)過(guò)程全部完成
// 關(guān)鍵點(diǎn)
a) 將數(shù)組中的每個(gè)元素?cái)?shù)組化,以便后續(xù)存放已經(jīng)排好序的元素
b) k的取值,k+=2, 每次取兩個(gè)數(shù)組進(jìn)行比較,形成一個(gè)新的數(shù)組
c) 一定要在比較完之后附加空數(shù)組,否則會(huì)在數(shù)組個(gè)數(shù)為奇數(shù)個(gè)的時(shí)候出現(xiàn)訪問(wèn)越界錯(cuò)誤
d) 最外層循環(huán)的lim的取值, lim = (lim + 1) / 2即原數(shù)組長(zhǎng)度的一半,作為內(nèi)循環(huán)終止的條件
*/
遞歸優(yōu)化,查表法-Memoization(記憶): 函數(shù)可以用對(duì)象去記住先前操縱的成果,從而能避免無(wú)謂的運(yùn)算
避免重復(fù)工作,將執(zhí)行過(guò)的運(yùn)算或操作,緩存起來(lái),如果后續(xù)有相同的操作可直接從緩存中查找,沒(méi)有則進(jìn)行遞歸,可大大減少遞歸的工作量,提高性能。
var count = 0;
function factorial(n) {
console.log("factorial count = " + count++);
if (n == 0) {
return 1;
} else {
return n * factorial(n - 1);
}
}
// var f1 = factorial(6);
// var f2 = factorial(5);
// var f3 = factorial(4);
// >>>>> 結(jié)果是函數(shù)被調(diào)用了:18次
/*
遞歸優(yōu)化:查表法,通過(guò)緩存
*/
function memFactorial(n) {
console.log("memFactorial count = " + count++);
// JS中函數(shù)即可視為一個(gè)對(duì)象,所以可以直接通過(guò)函數(shù)名+點(diǎn)語(yǔ)法給此對(duì)象添加屬性
if (!memFactorial.cache) {
memFactorial.cache = {
"0": 1,
"1": 1
};
}
// hasOwnProperty(n)可以判斷對(duì)象中是否存在該屬性,不會(huì)查找原型(但是"in"會(huì)先查實(shí)例再原型)
if (!memFactorial.cache.hasOwnProperty(n)) {
// 緩存中不存在的情況下再進(jìn)行遞歸,并且將新數(shù)組緩存起來(lái)
memFactorial.cache[n] = n * memFactorial(n - 1);
}
// 最終數(shù)據(jù)都可以在緩存中找到
return memFactorial.cache[n];
}
var f1 = memFactorial(6);
var f2 = memFactorial(5);
var f3 = memFactorial(4);
// >>>>> 結(jié)果是函數(shù)被調(diào)用了:只有8次,大大的減少了函數(shù)被調(diào)用的次數(shù)
Memoization通用版,但不建議使用,建議自己手動(dòng)在普通版中實(shí)現(xiàn)函數(shù)內(nèi)容
通用版需要緩存特定參數(shù)的函數(shù)調(diào)用結(jié)果,即,傳入的參數(shù)不同,調(diào)用函數(shù)的時(shí)候,其結(jié)果需要緩存起來(lái)
/*
遞歸優(yōu)化通用版,性能不如普通版,需要緩存每次調(diào)用結(jié)果, 即內(nèi)部函數(shù)
*/
function memoize(func, cache) {
// 緩存池
cache = cache || {}; // 沒(méi)有則新建
var result = function(arg) {
// console.log("memoize count = " + count++);
if (!cache.hasOwnProperty(arg)) {
cache[arg] = func(arg);
}
};
return result;
}
// 使用
// 將階乘函數(shù)緩存起來(lái)
// 只是優(yōu)化了cache的通用性,但損失了一部分性能
var memOpfactorial = memoize(factorial, {"0": 1, "1": 1});
var f1 = memOpfactorial(6);
var f2 = memOpfactorial(5);
var f3 = memOpfactorial(4);
// 結(jié)果次數(shù)依舊是:18次。因此不推薦使用
總結(jié)
- 謹(jǐn)慎使用遞歸,以防造成死循環(huán),程序崩潰,或者調(diào)用棧溢出;
- 學(xué)會(huì)使用迭代來(lái)替代遞歸,可以避免遞歸調(diào)用?;蛩姥h(huán)問(wèn)題出現(xiàn);
- 查表法,遞歸的優(yōu)化版:Memoization減少重復(fù)工作,提升性能。但不推薦使用通用版,相比普通版會(huì)損失部分性能。
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
- js中遞歸函數(shù)的使用介紹
- JavaScript的遞歸之遞歸與循環(huán)示例介紹
- JS 樹(shù)形遞歸實(shí)例代碼
- JavaScript Array Flatten 與遞歸使用介紹
- JavaScript 語(yǔ)言的遞歸編程
- JS中遞歸函數(shù)
- 總結(jié)javascript中的六種迭代器
- js數(shù)組的五種迭代方法及兩種歸并方法(推薦)
- javaScript數(shù)組迭代方法詳解
- JS的數(shù)組迭代方法
- 淺談javascript 迭代方法
- JavaScript中的迭代器和生成器詳解
- js 數(shù)組實(shí)現(xiàn)一個(gè)類(lèi)似ruby的迭代器
相關(guān)文章
JS實(shí)現(xiàn)單行文字不間斷向上滾動(dòng)的方法
這篇文章主要介紹了JS實(shí)現(xiàn)單行文字不間斷向上滾動(dòng)的方法,以實(shí)例形式較為詳細(xì)的分析了文字滾動(dòng)效果實(shí)現(xiàn)的原理與技巧,需要的朋友可以參考下2015-01-01
javascript限制文本框只允許輸入數(shù)字(曾經(jīng)與現(xiàn)在的方法對(duì)比)
很多時(shí)候需要用到限制文本框的數(shù)字輸入,試過(guò)許多方法,都不太理想,遂決定自己實(shí)現(xiàn)一個(gè)來(lái)玩玩,接下來(lái)介紹曾經(jīng)使用過(guò)的方法與自定義方法的對(duì)比,感興趣的朋友可以了解下啊2013-01-01
純js實(shí)現(xiàn)瀑布流布局及ajax動(dòng)態(tài)新增數(shù)據(jù)
這篇文章主要介紹了基于javascript實(shí)現(xiàn)瀑布流布局,及ajax動(dòng)態(tài)新增數(shù)據(jù)的相關(guān)資料,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-04-04
分享十八個(gè)殺手級(jí)JavaScript單行代碼
這篇文章主要給大家分享了十八個(gè)殺手級(jí)JavaScript單行代碼,這些單行代碼可以幫助你提高工作效率并可以幫助調(diào)試代碼,對(duì)大家學(xué)習(xí)或者使用JavaScript具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2021-10-10
javascript使用正則表達(dá)式實(shí)現(xiàn)注冊(cè)登入校驗(yàn)
這篇文章主要為大家詳細(xì)介紹了javascript使用正則表達(dá)式實(shí)現(xiàn)注冊(cè)登入校驗(yàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-09-09
js實(shí)現(xiàn)類(lèi)似于add(1)(2)(3)調(diào)用方式的方法
這篇文章主要介紹了js實(shí)現(xiàn)類(lèi)似于add(1)(2)(3)調(diào)用方式的方法,需要的朋友可以參考下2015-03-03
layui實(shí)現(xiàn)左側(cè)菜單點(diǎn)擊右側(cè)內(nèi)容區(qū)顯示
這篇文章主要為大家詳細(xì)介紹了layui實(shí)現(xiàn)左側(cè)菜單點(diǎn)擊右側(cè)內(nèi)容區(qū)顯示,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-07-07
JavaScript中AOP的實(shí)現(xiàn)與應(yīng)用
這篇文章主要給大家介紹了關(guān)于JavaScript中AOP的實(shí)現(xiàn)與應(yīng)用的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用JavaScript具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-05-05
javascript實(shí)現(xiàn)計(jì)算指定范圍內(nèi)的質(zhì)數(shù)示例
這篇文章主要介紹了javascript實(shí)現(xiàn)計(jì)算指定范圍內(nèi)的質(zhì)數(shù),涉及javascript數(shù)值計(jì)算與判斷相關(guān)操作技巧,需要的朋友可以參考下2018-12-12
javascript實(shí)現(xiàn)別踩白塊兒小游戲程序
我們先看到的未開(kāi)始的頁(yè)面黑色和白色方塊是隨機(jī)生成的,完了這么多把沒(méi)有發(fā)現(xiàn)時(shí)固定不變的。點(diǎn)擊一次方塊,程序需要判斷是黑色還是白色的。如果是黑色的,當(dāng)然程序也是實(shí)現(xiàn)了一次自減,在動(dòng)畫(huà)中是實(shí)現(xiàn)一次下移的,下移的時(shí)候點(diǎn)擊的方塊到黃顏色的區(qū)域會(huì)變成灰色2015-11-11

