什麼是遞歸演算法
Ⅰ 遞歸是什麼意思
程序調用自身的編程技巧稱為遞歸( recursion)。
構成遞歸需具備的條件有:
1、子問題須與原始問題為同樣的事,且更為簡單。
2、不能無限制地調用本身,須有個出口,化簡為非遞歸狀況處理。
遞歸做為一種演算法在程序設計語言中廣泛應用。 一個過程或函數在其定義或說明中有直接或間接調用自身的一種方法,它通常把一個大型復雜的問題層層轉化為一個與原問題相似的規模較小的問題來求解,遞歸策略只需少量的程序就可描述出解題過程所需要的多次重復計算,大大地減少了程序的代碼量。
(1)什麼是遞歸演算法擴展閱讀:
遞歸一般用於解決三類問題:
1、數據的定義是按遞歸定義的。(Fibonacci函數,n的階乘)
2、問題解法按遞歸實現。(回溯)
3、數據的結構形式是按遞歸定義的。(二叉樹的遍歷,圖的搜索)
遞歸的缺點:
遞歸解題相對常用的演算法如普通循環等,運行效率較低。因此,應該盡量避免使用遞歸,除非沒有更好的演算法或者某種特定情況,遞歸更為適合的時候。在遞歸調用的過程當中系統為每一層的返回點、局部量等開辟了棧來存儲,因此遞歸次數過多容易造成棧溢出。
Ⅱ 漢諾塔遞歸演算法是什麼
hanot (n-1,b,a,c);(解釋:在把B塔上的(n-1))個藉助A塔移動到C塔)
為了實現 n個盤從 藉助c 從a 移動到 b
思路如下:
首先考慮極限當只有一個盤的時候,盤直接從 a -> b即可。
當有2個盤的時候,把1號盤從a -> c 然後 把2號盤 a->b 再 把 2好盤從 c - > b。
當有n個盤的時候,把 n-1個 盤 藉助 b 移動到 c 然後將 n號盤從 a -> b。
這時候只要將 n-1想辦法從c移動到 b 藉助 a 那麼就可以先把 n-2個盤藉助b移動到a。
遞歸,就是在運行的過程中調用自己。
構成遞歸需具備的條件:
1,子問題須與原始問題為同樣的事,且更為簡單;
2,不能無限制地調用本身,須有個出口,化簡為非遞歸狀況處理。
在數學和計算機科學中,遞歸指由一種(或多種)簡單的基本情況定義的一類對象或方法,並規定其他所有情況都能被還原為其基本情況。
以上內容參考:網路-遞歸公式
Ⅲ 什麼叫遞歸法
1、遞歸演算法概念:
在函數或子過程的內部,直接或者間接地調用自己的演算法。
2、基本信息:
遞歸演算法是把問題轉化為規模縮小了的同類問題的子問題。然後遞歸調用函數或過程來表示問題的解。一個過程或函數直接或間接調用自己本身,這種過程或函數叫遞歸過程或函數。
Ⅳ 什麼是遞歸演算法,演算法學習哪個網站好呢
遞歸演算法指的是函數/過程/子程序在運行過程中直接或間接調用自身而產生的重入現象。看過馬克威演算法交易平台,裡面涵蓋開源的演算法以及馬克威演算法,另外還有機器學習等內容,真心好,我還下載了幾個演算法研究了下.....頗合我意~希望可以幫到你。
Ⅳ 遞歸演算法是什麼
遞歸演算法(英語:recursion algorithm)在計算機科學中是指一種通過重復將問題分解為同類的子問題而解決問題的方法。
遞歸式方法可以被用於解決很多的計算機科學問題,因此它是計算機科學中十分重要的一個概念。絕大多數編程語言支持函數的自調用,在這些語言中函數可以通過調用自身來進行遞歸。
計算理論可以證明遞歸的作用可以完全取代循環,因此在很多函數編程語言(如Scheme)中習慣用遞歸來實現循環。
Ⅵ 什麼是遞歸演算法
遞歸演算法就是一個函數通過不斷對自己的調用而求得最終結果的一種思維巧妙但是開銷很大的演算法。
比如:
漢諾塔的遞歸演算法:
void move(char x,char y){
printf("%c-->%c\n",x,y);
}
void hanoi(int n,char one,char two,char three){
/*將n個盤從one座藉助two座,移到three座*/
if(n==1) move(one,three);
else{
hanoi(n-1,one,three,two);
move(one,three);
hanoi(n-1,two,one,three);
}
}
main(){
int n;
printf("input the number of diskes:");
scanf("%d",&n);
printf("The step to moving %3d diskes:\n",n);
hanoi(n,'A','B','C');
}
我說下遞歸的理解方法
首先:對於遞歸這一類函數,你不要糾結於他是干什麼的,只要知道他的一個模糊功能是什麼就行,等於把他想像成一個能實現某項功能的黑盒子,而不去管它的內部操作先,好,我們來看下漢諾塔是怎麼樣解決的
首先按我上面說的把遞歸函數想像成某個功能的黑盒子,void hanoi(int n,char one,char two,char three); 這個遞歸函數的功能是:能將n個由小到大放置的小長方形從one 位置,經過two位置 移動到three位置。那麼你的主程序要解決的問題是要將m個的"漢諾塊"由A藉助B移動到C,根據我們上面說的漢諾塔的功能,我相信傻子也知道在主函數中寫道:hanoi(m,A,B,C)就能實現將m個塊由A藉助B碼放到C,對吧?所以,mian函數裡面有hanoi(m,'A','C','B');這個調用。
接下來我們看看要實現hannoi的這個功能,hannoi函數應該幹些什麼?
在hannoi函數里有這么三行
hanoi(n-1,one,three,two);
move(one,three);
hanoi(n-1,two,one,three);
同樣以黑盒子的思想看待他,要想把n個塊由A經過B搬到C去,是不是可以分為上面三步呢?
這三部是:第一步將除了最後最長的那一塊以外的n-1塊由one位置經由three搬到two 也就是從A由C搬到B 然後把最下面最長那一塊用move函數把他從A直接搬到C 完事後 第三步再次將剛剛的n-1塊藉助hannoi函數的功能從B由A搬回到C 這樣的三步實習了n塊由A經過B到C這樣一個功能,同樣你不用糾結於hanoi函數到底如何實現這個功能的,只要知道他有這么一個神奇的功能就行
最後:遞歸都有收尾的時候對吧,收尾就是當只有一塊的時候漢諾塔怎麼個玩法呢?很簡單吧,直接把那一塊有Amove到C我們就完成了,所以hanoni這個函數最後還要加上 if(n==1)move(one,three);(當只有一塊時,直接有Amove到C位置就行)這么一個條件就能實現hanoin函數n>=1時將n個塊由A經由B搬到C的完整功能了。
遞歸這個復雜的思想就是這樣簡單解決的,呵呵 不知道你看懂沒?純手打,希望能幫你理解遞歸
總結起來就是不要管遞歸的具體實現細節步驟,只要知道他的功能是什麼,然後利用他自己的功能通過調用他自己去解決自己的功能(好繞口啊,日)最後加上一個極限情況的條件即可,比如上面說的1個的情況。
Ⅶ 遞歸演算法的是怎麼回事
和迭代差不多,只是通過定義和調用函數來實現迭代
把事情分解成相同的步驟重復執行直到符合某一條件時結束,再反過來遞推到最初的狀態,問題就解決了
比如定義(用的是C語言)
int fun(int a)
{
if(a==1) return 1;
else
{
a=a*fun(a-1);
return a;
}
}
在fun裡面再定義fun,這個fun都只做一件事,把a的內容和fun(a-1)相乘作為返回值
這里要有個終止條件,即a=1時返回值為1,這樣,如果我給最初的fun里的a賦值為5,第一步為5*fun(4),而執行fun(4)的結果為4*fun(3)....直到fun(2)=2*fun(1)即fun(2)=2*1,再把fun(2)代回去,得fun(3)=3*2*1,最後倒推的結果為fun(5)=5*4*3*2*1,即這個遞歸函數實現了a的階乘fun(a)=a!
夠詳細了吧,覺得好的話給我加分吧 ^_^
Ⅷ 什麼是遞歸演算法有什麼作用
作者名:不詳 來源:網友提供 05年7月7日
無法貼圖 ,自己到 http://51zk.csai.cn/sjjg/NO00223.htm 看去吧
1、調用子程序的含義:
在過程和函數的學習中,我們知道調用子程序的一般形式是:主程序調用子程序A,子程序A調用子程序B,如圖如示,這個過程實際上是:
@當主程序執行到調用子程序A語句時,系統保存一些必要的現場數據,然後執行類似於BASIC語言的GOTO語句,跳轉到子程序A(為了說得簡單些,我這里忽略了參數傳遞這個過程)。
@當子程序A執行到調用子程序B語句時,系統作法如上,跳轉到子程序B。
@子程序B執行完所有語句後,跳轉回子程序A調用子程序B語句的下一條語句(我這又忽略了返回值處理)
@子程序A執行完後,跳轉回主程序調用子程序A語句的下一條語句
@主程序執行到結束。
做個比較:我在吃飯(執行主程序)吃到一半時,某人叫我(執行子程序A),話正說到一半,電話又響了起來(執行子程序B),我只要先接完電話,再和某人把話說完,最後把飯吃完(我這飯吃得也夠累的了J)。
2、認識遞歸函數
我們在高中時都學過數學歸納法,例:
求 n!
我們可以把n!這么定義
也就是說要求3!,我們必須先求出2!,要求2!,必須先求1!,要求1!,就必須先求0!,而0!=1,所以1!=0!*1=1,再進而求2!,3!。分別用函數表示,則如圖:
我們可以觀察到,除計算0!子程序外,其他的子程序基本相似,我們可以設計這么一個子程序:
int factorial(int i){
int res;
res=factorial(I-1)*i;
return res;
}
那麼當執行主程序語句s=factorial(3)時,就會執行factorial(3),但在執行factorial(3),又會調用factorial(2),這時大家要注意,factorial(3)和factorial(2)雖然是同一個代碼段,但在內存中它的數據區是兩份!而執行factorial(2)時又會調用factorial(1),執行factorial(1)時又會調用factorial(0),每調用一次factorial函數,它就會在內存中新增一個數據區,那麼這些復制了多份的函數大家可以把它看成是多個不同名的函數來理解;
但我們這個函數有點問題,在執行factorial(0)時,它又會調用factorial(-1)。。。造成死循環,也就是說,在factorial函數中,我們要在適當的時候保證不再調用該函數,也就是不執行res=factorial(I-1)*i;這條調用語句。所以函數要改成:
int factorial(int i){
int res;
if (I>0) res=factorial(I-1)*i; else res=1;
return res;
}
那麼求3!的實際執行過程如圖所示:
3、如何考慮用遞歸的方法來解決問題
例:求s=1+2+3+4+5+6+……+n
本來這個問題我們過去常用循環累加的方法。而這里如要用遞歸的方法,必須考慮兩點:
1) 能否把問題轉化成遞歸形式的描述;
2) 是否有遞歸結束的邊界條件。
設:函數s(n)=1+2+3+…+(n-1)+n
顯然遞歸的兩個條件都有了:
1) s(n) =s(n-1)+n
2) s(1)=1
所以源程序為:
int progression(int n){
int res;
if (n=1 )res=1 else res=progression(n-1)+n;
return res;
}
4、遞歸的應用
中序遍歷二叉樹
void inorder (BinTree T){
if (T){
inorder(T->lchild);
printf(「%c」,T->data);
inorder(T->rchild);
}
}
現假設樹如圖(為了講解方便,樹很簡單)
@執行第一次調用inorder1,T指向頂結點,T不為空,所以第二次調用inorder2;
@T指向頂結點的左子樹結點也就是B,不為空,所以第三次調用inorder3;
@T指向B結點的左子樹結點,為空,所以什麼都不執行,返回inorder2;
@列印B結點的DATA域值「b」;
@第四次調用inorder4,去訪問B子樹的右結點
@T指向B結點的右子樹結點,為空,所以什麼都不執行,返回inorder2;
@返回inorder1;
@列印A結點的DATA域值「a」;
@第五次調用inorder5,去訪問A子樹的右結點;
@T指向A結點的右子樹結點,為空,所以什麼都不執行,返回inorder1;
@inorder1執行完畢,返回。