摘要:前言一般地對于語言而言普通的遞歸調(diào)用是在虛擬機(jī)棧上完成的假如是一個遞歸方法那么在其內(nèi)部再調(diào)用自己的時候假設(shè)為那么方法變量表將創(chuàng)建在方法棧幀之上從而形成了一個新的棧幀因此容易發(fā)現(xiàn)在遞歸思想中遞歸簡化了問題的表達(dá)但犧牲了虛擬機(jī)棧中的內(nèi)存空間普通
前言
一般地,對于java語言而言,普通的遞歸調(diào)用是在java虛擬機(jī)棧上完成的.假如a()是一個遞歸方法,那么在其內(nèi)部再調(diào)用自己的時候,假設(shè)為a1(),那么a1()方法變量表將創(chuàng)建在a()方法棧幀之上,從而形成了一個新的棧幀.因此容易發(fā)現(xiàn),在遞歸思想中,遞歸簡化了問題的表達(dá),但犧牲了虛擬機(jī)棧中的內(nèi)存空間.
普通遞歸 斐波那契遞歸法public static int fib(int num){ if(num<2) return num; else return fib(num-2)+fib(num-1); }
對于上面的解法,很容易就會發(fā)現(xiàn),不但屬于普通遞歸,而且在計(jì)算fib(num-1)是重復(fù)了fib(num-2)的計(jì)算量,因此代碼效率大打折扣.因此效率較高的寫法可以用for循環(huán)計(jì)算,
public static int fib3(int n) { if (n < 2) return n; else { int pre = 0; int suf = 1; for (int i = 2; i <= n; i++) { int temp = suf; suf += pre; pre = temp; } return suf; } }斐波那契尾遞歸優(yōu)化
public class Main { public static void main(String[] args) { System.out.print(fib2(3, 0, 1)); } public static int fib2(int count, int pre, int result) { if (count == 1) return result; else return fib2(--count, result, result + pre); } }性能對比
public static void main(String[] args) { long time = new Date().getTime(); int num=40; System.out.println(fib(num)); System.out.println("普通遞歸調(diào)用用時:" + (new Date().getTime() - time) + "毫秒"); time = new Date().getTime(); System.out.println(fib2(num, 0, 1)); System.out.println("尾遞歸優(yōu)化調(diào)用用時:" + (new Date().getTime() - time) + "毫秒"); time = new Date().getTime(); System.out.println(fib3(num)); System.out.println("for循環(huán)法調(diào)用用時:" + (new Date().getTime() - time) + "毫秒"); } //輸出 /* 102334155 普通遞歸調(diào)用用時:674毫秒 102334155 尾遞歸優(yōu)化調(diào)用用時:0毫秒 102334155 for循環(huán)法調(diào)用用時:0毫秒 */
可以看出有明顯差異,即使普通遞歸法計(jì)算量多了一半,時間除以2也是387毫秒,這也遠(yuǎn)遠(yuǎn)高于for循環(huán)和遞歸尾優(yōu)化法.
尾遞歸優(yōu)化思想即遞歸方法return 直接返回方法,注意是直接返回方法,不能是方法加1個值等形式.這樣在遞歸調(diào)用時,新方法會覆蓋當(dāng)前棧幀,達(dá)到節(jié)省??臻g的目的.因此也就不會有遞歸調(diào)用產(chǎn)生的棧溢出問題.
尾遞歸寫法//count作為計(jì)數(shù),表示遞歸層次, //pre代表前一個值 //result 表示當(dāng)前值 public static int fib2(int count, int pre, int result) { //層次減到1時返回計(jì)算結(jié)果 if (count == 1) return result; else{ //遞歸調(diào)用時,層次減1,前一項(xiàng)更新為當(dāng)前項(xiàng),所以填result,第三個參數(shù)即實(shí)現(xiàn)了倒數(shù)第二個參數(shù)加倒數(shù)第一個參數(shù). return fib2(--count, result, result + pre); } }
總體而言參數(shù)的書寫分為兩部分
前部分為計(jì)數(shù),后部分為計(jì)算,例如計(jì)算階乘時候只需要兩個參數(shù),第一個計(jì)數(shù),第二個存結(jié)果.
尾遞歸將全部信息放入了參數(shù)里,因此也就巧妙地避免了需要上一棧幀保存信息.
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/43320.html
摘要:前言一般地對于語言而言普通的遞歸調(diào)用是在虛擬機(jī)棧上完成的假如是一個遞歸方法那么在其內(nèi)部再調(diào)用自己的時候假設(shè)為那么方法變量表將創(chuàng)建在方法棧幀之上從而形成了一個新的棧幀因此容易發(fā)現(xiàn)在遞歸思想中遞歸簡化了問題的表達(dá)但犧牲了虛擬機(jī)棧中的內(nèi)存空間普通 前言 一般地,對于java語言而言,普通的遞歸調(diào)用是在java虛擬機(jī)棧上完成的.假如a()是一個遞歸方法,那么在其內(nèi)部再調(diào)用自己的時候,假設(shè)為a1...
摘要:前言一般地對于語言而言普通的遞歸調(diào)用是在虛擬機(jī)棧上完成的假如是一個遞歸方法那么在其內(nèi)部再調(diào)用自己的時候假設(shè)為那么方法變量表將創(chuàng)建在方法棧幀之上從而形成了一個新的棧幀因此容易發(fā)現(xiàn)在遞歸思想中遞歸簡化了問題的表達(dá)但犧牲了虛擬機(jī)棧中的內(nèi)存空間普通 前言 一般地,對于java語言而言,普通的遞歸調(diào)用是在java虛擬機(jī)棧上完成的.假如a()是一個遞歸方法,那么在其內(nèi)部再調(diào)用自己的時候,假設(shè)為a1...
摘要:尾遞歸優(yōu)化一般遞歸與尾遞歸一般遞歸執(zhí)行可以看到一般遞歸每一級遞歸都產(chǎn)生了新的局部變量必須創(chuàng)建新的調(diào)用棧隨著遞歸深度的增加創(chuàng)建的棧越來越多造成爆棧尾遞歸尾遞歸基于函數(shù)的尾調(diào)用每一級調(diào)用直接返回遞歸函數(shù)更新調(diào)用棧沒有新局部變量的產(chǎn)生類似迭代的 Python尾遞歸優(yōu)化 一般遞歸與尾遞歸 一般遞歸: def normal_recursion(n): if n == 1: ...
摘要:執(zhí)行完了,銷毀調(diào)用棧中自己的記錄,依次銷毀和的調(diào)用幀,最后完成整個流程。尾遞歸定義先來看一下遞歸,當(dāng)一個函數(shù)調(diào)用自身,就叫做遞歸。 尾調(diào)用 1. 定義 尾調(diào)用是函數(shù)式編程中一個很重要的概念,當(dāng)一個函數(shù)執(zhí)行時的最后一個步驟是返回另一個函數(shù)的調(diào)用,這就叫做尾調(diào)用。 注意這里函數(shù)的調(diào)用方式是無所謂的,以下方式均可: 函數(shù)調(diào)用: func(···) 方法調(diào)用: obj.meth...
摘要:根據(jù)該規(guī)則,返回第個斐波那契數(shù)。尾遞歸函數(shù)調(diào)用自身,稱為遞歸。一個前端眼中的斐波那契數(shù)列解斐波那契數(shù)列的實(shí)用解法調(diào)用棧尾遞歸和手動優(yōu)化尾調(diào)用優(yōu)化譯我從用寫斐波那契生成器中學(xué)到的令人驚訝的件事 斐波那契數(shù)列是以下一系列數(shù)字: 0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233, 377, 610, 987, ... 在種子數(shù)字 0 和 1 ...
閱讀 2220·2021-11-19 09:40
閱讀 1932·2021-11-08 13:24
閱讀 2463·2021-10-18 13:24
閱讀 2868·2021-10-11 10:57
閱讀 3592·2021-09-22 15:42
閱讀 1127·2019-08-29 17:11
閱讀 2538·2019-08-29 16:11
閱讀 2430·2019-08-29 11:11