軟體開發(fā)之遞迴操作
Aug 16, 2024 pm 07:54 PM我們來看看這個經(jīng)典的遞歸階乘:
#include int factorial(int n) { int previous = 0xdeadbeef; if (n == 0 || n == 1) { return 1; } previous = factorial(n-1); return n * previous; } int main(int argc) { int answer = factorial(5); printf("%d\n", answer); }
遞歸階乘 - factorial.c
函數(shù)呼叫自身的這個觀點在一開始是讓人很難理解的。為了讓這個過程更具體,下圖展示的是當呼叫 factorial(5) 並且達到 n == 1這行程式碼 時,堆疊上 端點的情況:
每次呼叫 factorial 都會產(chǎn)生一個新的 堆疊幀。這些堆疊幀的創(chuàng)建和 銷毀 是使得遞歸版本的階乘慢於其對應(yīng)的迭代版本的原因。在呼叫返回之前,累積的這些堆疊幀可能會耗盡堆疊空間,進而使你的程式崩潰。
而這些擔憂經(jīng)常是存在於理論上的。例如,對於每個 factorial 的堆疊幀佔用 16 位元組(這可能取決於堆疊排列以及其它因素)。如果在你的電腦上運行著現(xiàn)代的 x86 的 Linux 內(nèi)核,一般情況下你擁有 8 GB 的??臻g,因此,factorial 程式中的 n 最多可以達到 512,000 左右。這是一個巨大無比的結(jié)果,它將花費8,971,833 位元來表示這個結(jié)果,因此,棧空間根本就不是什麼問題:一個極小的整數(shù)—— 甚至是一個64 位元的整數(shù)—— 在我們的堆疊空間被耗盡之前就早已經(jīng)溢出了成千上萬次了。
過一會兒我們再去看 CPU 的使用,現(xiàn)在,我們先從位元和位元組回退一步,把遞歸看成一種通用技術(shù)。我們的階乘演算法可歸結(jié)為:將整數(shù) N、N-1、 … 1 推入到一個棧,然後將它們以相反的順序相乘。實際上我們使用了程式呼叫堆疊來實現(xiàn)這一點,這是它的細節(jié):我們在堆上分配一個堆疊並使用它。雖然呼叫棧具有特殊的特性,但是它也只是另一個資料結(jié)構(gòu)而已,你可以隨意使用。我希望這個示意圖可以讓你明白這一點。
當你將堆疊呼叫視為一種資料結(jié)構(gòu),有些事情將變得更加清晰明了:將那些整數(shù)堆積起來,然後再將它們相乘,這並不是一個好的想法。那是一種有缺陷的實現(xiàn):就像你拿螺絲起子去釘釘子一樣。相對較合理的是使用一個迭代過程去計算階乘。
但是,螺絲釘太多了,我們只能挑一個。有一個經(jīng)典的面試題,在迷宮裡有一隻老鼠,你必須幫助這隻老鼠找到一個起司。假設(shè)老鼠能夠在迷宮中向左或向右轉(zhuǎn)。你該怎麼去建模來解決這個問題?
就像現(xiàn)實生活中的許多問題一樣,你可以將這個老鼠找起司的問題簡化為一個圖,一個二叉樹的每個結(jié)點代表在迷宮中的一個位置。然後你可以讓老鼠在任何可能的地方都左轉(zhuǎn),而當它進入一個死胡同時,再回溯回去,再右轉(zhuǎn)。這是一個老鼠行走的 迷宮範例:
每到邊緣(線)都讓老鼠左轉(zhuǎn)或右轉(zhuǎn)到達一個新的位置。如果向哪邊轉(zhuǎn)都被攔住,表示相關(guān)的邊緣不存在。現(xiàn)在,我們來討論一下!這個過程無論你是呼叫堆疊還是其它資料結(jié)構(gòu),它都離不開一個遞歸的過程。而使用呼叫棧是非常容易的:
#include #include "maze.h" int explore(maze_t *node) { int found = 0; if (node == NULL) { return 0; } if (node->hasCheese){ return 1;// found cheese } found = explore(node->left) || explore(node->right); return found; } int main(int argc) { int found = explore(&maze); }
遞歸迷宮解法 下載
當我們在 maze.c:13 中找到起司時,堆疊的情況如下圖所示。你也可以在 GDB 輸出 中看到更詳細的數(shù)據(jù),它是使用 指令 收集的數(shù)據(jù)。
它展示了遞歸的良好表現(xiàn),因為這是一個適合使用遞歸的問題。而且這並不奇怪:當涉及到演算法時,遞歸是規(guī)則,而不是例外。它出現(xiàn)在如下情景中——進行搜尋時、進行遍歷樹和其它資料結(jié)構(gòu)時、進行解析時、需要排序時——它無所不在。正如眾所周知的 pi 或 e,它們在數(shù)學中像「神」一樣的存在,因為它們是宇宙萬物的基礎(chǔ),而遞歸也和它們一樣:只是它存在於計算結(jié)構(gòu)中。
Steven Skienna 的優(yōu)秀著作 演算法設(shè)計指南 的精彩之處在於,他透過 「戰(zhàn)爭故事」 作為手段來詮釋工作,以此來展示解決現(xiàn)實世界中的問題背後的演算法。這是我所知道的拓展你的演算法知識的最佳資源。另一個讀物是 McCarthy 的 關(guān)於 LISP 實現(xiàn)的的原創(chuàng)論文。遞歸在語言中既是它的名字也是它的基本原理。這篇論文既可讀又有趣,在工作中能看到大師的作品是件讓人興奮的事。
Back to the maze problem. Although it is difficult to leave recursion here, it does not mean that it must be achieved through the call stack. You can use a string like RRLL to track the turn, and then use this string to determine the mouse's next move. Or you could assign something else to record the entire status of the cheese hunt. You still implement a recursive process, you just need to implement your own data structure.
That seems more complicated because stack calls are more appropriate. Each stack frame records not only the current node, but also the state of the computation on that node (in this case, whether we only let it go to the left, or have tried to go to the right). Therefore, the code has become unimportant. However, sometimes we abandon this excellent algorithm because of fear of overflow and expected performance. That's stupid!
As we can see, the stack space is very large, and other limitations are often encountered before the stack space is exhausted. On the one hand, you can check the size of the problem to ensure that it can be handled safely. The CPU concerns are driven by two widely circulated problematic examples: dumb factorial and the horrific memoryless O(2n) Fibonacci recursion. They are not correct representations of stack recursive algorithms.
In fact stack operations are very fast. Typically, the stack offset to data is very accurate, it is hot data in the cache, and specialized instructions operate on it. At the same time, the overhead associated with using your own data structures allocated on the heap is significant. It is often seen that people write implementation methods that are more complex and have worse performance than stack call recursion. Finally, the performance of modern CPUs is very good, and generally the CPU will not be the performance bottleneck. Be careful when considering sacrificing program simplicity, just as you always consider program performance and the measurement of that performance.
The next article will be the last one in the Exploring Stack series. We will learn about tail calls, closures, and other related concepts. Then, it's time to dive into our old friend, the Linux kernel. Thank you for reading!
以上是軟體開發(fā)之遞迴操作的詳細內(nèi)容。更多資訊請關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

熱AI工具

Undress AI Tool
免費脫衣圖片

Undresser.AI Undress
人工智慧驅(qū)動的應(yīng)用程序,用於創(chuàng)建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發(fā)環(huán)境

Dreamweaver CS6
視覺化網(wǎng)頁開發(fā)工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

在CentOS上集成Postman應(yīng)用可以通過多種方法來實現(xiàn),以下是詳細的步驟和建議:通過下載安裝包安裝Postman下載Postman的Linux版本安裝包:訪問Postman官方網(wǎng)站,選擇適用於Linux的版本進行下載。解壓安裝包:使用以下命令將安裝包解壓到指定目錄,例如/opt:sudotar-xzfpostman-linux-x64-xx.xx.xx.tar.gz-C/opt請注意將“postman-linux-x64-xx.xx.xx.tar.gz”替換為您實際下載的文件名。創(chuàng)建符號

Java與其他編程語言的主要區(qū)別在於其“一次編寫,到處運行”的跨平臺特性。 1.Java的語法接近C ,但去掉了容易出錯的指針操作,適合大型企業(yè)應(yīng)用。 2.與Python相比,Java在性能和大規(guī)模數(shù)據(jù)處理上更具優(yōu)勢。 Java的跨平臺優(yōu)勢源於Java虛擬機(JVM),它能在不同平臺上運行相同的字節(jié)碼,簡化開發(fā)和部署,但需注意避免使用平臺特定API以保持跨平臺性。

在PyCharm中設(shè)置解釋器的位置可以通過以下步驟實現(xiàn):1.打開PyCharm,點擊“File”菜單,選擇“Settings”或“Preferences”。 2.找到並點擊“Project:[你的項目名]”,然後選擇“PythonInterpreter”。 3.點擊“AddInterpreter”,選擇“SystemInterpreter”,瀏覽到Python安裝目錄,選中Python可執(zhí)行文件,點擊“OK”。設(shè)置解釋器時需注意路徑正確性、版本兼容性和虛擬環(huán)境的使用,以確保項目順利運行。

在VSCode中手動安裝插件包的步驟是:1.下載插件的.vsix文件;2.打開VSCode並按Ctrl Shift P(Windows/Linux)或Cmd Shift P(Mac)調(diào)出命令面板;3.輸入並選擇Extensions:InstallfromVSIX...,然後選擇.vsix文件並安裝。手動安裝插件提供了一種靈活的安裝方式,特別是在網(wǎng)絡(luò)受限或插件市場不可用時,但需要注意文件安全和可能的依賴問題。

【常見目錄說明】目錄/bin存放二進制可執(zhí)行文件(ls,cat,mkdir等),常用命令一般都在這裡。 /etc存放系統(tǒng)管理和配置文件/home存放所有用戶文件的根目錄,是用戶主目錄的基點,比如用戶user的主目錄就是/home/user,可以用~user表示/usr用於存放系統(tǒng)應(yīng)用程序,比較重要的目錄/usr/local?本地系統(tǒng)管理員軟件安裝目錄(安裝系統(tǒng)級的應(yīng)用)。這是最龐大的目錄,要用到的應(yīng)用程序和文件幾乎都在這個目錄。 /usr/x11r6?存放x?window的目錄/usr/bin?眾多

了解Nginx的配置文件路徑和初始設(shè)置非常重要,因為它是優(yōu)化和管理Web服務(wù)器的第一步。 1)配置文件路徑通常是/etc/nginx/nginx.conf,使用nginx-t命令可以查找並測試語法。 2)初始設(shè)置包括全局設(shè)置(如user、worker_processes)和HTTP設(shè)置(如include、log_format),這些設(shè)置允許根據(jù)需求進行定制和擴展,錯誤配置可能導致性能問題和安全漏洞。

MySQL的安裝和配置可以通過以下步驟完成:1.從官方網(wǎng)站下載適合操作系統(tǒng)的安裝包。 2.運行安裝程序,選擇“開發(fā)者默認”選項並設(shè)置root用戶密碼。 3.安裝後配置環(huán)境變量,確保MySQL的bin目錄在PATH中。 4.創(chuàng)建用戶時遵循最小權(quán)限原則並設(shè)置強密碼。 5.優(yōu)化性能時調(diào)整innodb_buffer_pool_size和max_connections參數(shù)。 6.定期備份數(shù)據(jù)庫並優(yōu)化查詢語句以提高性能。

Informix和MySQL均為廣受青睞的關(guān)係型數(shù)據(jù)庫管理系統(tǒng),它們在Linux環(huán)境下均表現(xiàn)優(yōu)異並得到廣泛應(yīng)用。以下是對兩者在Linux平臺上的對比分析:安裝與配置Informix:在Linux上部署Informix需要下載對應(yīng)的安裝文件,隨後依據(jù)官方文檔指引完成安裝及配置流程。 MySQL:MySQL的安裝過程較為簡便,可通過系統(tǒng)的包管理工具(例如apt或yum)輕鬆實現(xiàn)安裝,並且網(wǎng)絡(luò)上有大量的教程和社區(qū)支持可供參考。性能表現(xiàn)Informix:Informix以卓越的性能和
