• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            S.l.e!ep.¢%

            像打了激速一樣,以四倍的速度運轉,開心的工作
            簡單、開放、平等的公司文化;尊重個性、自由與個人價值;
            posts - 1098, comments - 335, trackbacks - 0, articles - 1
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

            小覽call stack(調用棧) (一)

            Posted on 2009-11-02 17:30 S.l.e!ep.¢% 閱讀(817) 評論(1)  編輯 收藏 引用 所屬分類: WinDbg

            棧在計算機領域中是個經常提到的名詞,數據結構中有棧;網絡傳輸中有協議棧。今天我們討論的調用棧(call stack),指的是在程序的執行過程中存儲函數調用信息的動態數據結構。

            ?

            這個定義可能太抽象了一些,在給出具體的例子之前,請大家先思考一個問題,哪些信息是函數調用過程中所需要的?或者這么問,一個編譯器,在面對一個函數的調用指令時,該生成哪些代碼?

            ?

            首先,函數的返回地址要保存下來。就好像你和你的小狗玩仍飛碟游戲,每一個函數調用好比扔一個飛碟,當你的狗狗哼茲哼茲的撿來飛碟,函數完執行的時候,它一定得知道去哪里把飛碟還給你。

            ?

            然后,函數的參數是個必不可少的元素,這個很直觀,就不多羅嗦了。第三,被調用的函數的局部變量也要存儲在棧上。因為根據局部標量的定義,對相同函數的不同調用,局部變量有不同的存儲空間,不會互相影響,所以這些數據也是跟函數調用息息相關的信息。

            ?

            下面,我們通過一個例子,來看看函數的調用棧中的信息:

            對于下面一段c++程序

            view plaincopy to clipboardprint?
            #include <stdio.h>??
            ?
            int SumFromOne(int d)??
            {??
            ??? int sum = 0xabcd;??
            ??? if (d == 1)??
            ??????? sum = 1;??
            ??? else???
            ??????? sum = d + SumFromOne(d-1);??
            ??? return sum;??
            }??
            ?
            void main()??
            {??
            ??? int sum = SumFromOne(10);??
            ??? printf("sum=%d", sum);??
            }?
            #include <stdio.h>

            int SumFromOne(int d)
            {
            ??? int sum = 0xabcd;
            ??? if (d == 1)
            ??????? sum = 1;
            ??? else
            ??????? sum = d + SumFromOne(d-1);
            ??? return sum;
            }

            void main()
            {
            ??? int sum = SumFromOne(10);
            ??? printf("sum=%d", sum);
            }
            ?

            ?

            編譯之,Cl /Zi a.cpp (/Zi生成pdb,調試的時候使用)


            大家選用熟悉的調試器,在這里,筆者用的是windbg 大家可以去這個地址下載(http://www.microsoft.com/whdc/devtools/debugging/installx86.Mspx)


            從調試器中啟動程序:Windbg a.exe

            然后在第4行設置一個斷點(F9)。開始執行這個程序(F5),直到程序中斷在斷點處


            找到程序的調用棧:

            1.?????? 察看當前的ebp,在command窗口中應該已經看到。否則的話,在command中輸入r

            2.?????? 在memory察看窗口中,virtual欄中輸入ebp-10的值,并且把display format改成long hex,以利于觀察棧中的值


            我把我的windbg截圖粘貼如下,并和大家一起觀察幾個地方

            ?

            1.?????? 返回地址0040106b。參見反匯編的結果,0040106b正是main調完SumFromOne之后的那條指令。

            2.?????? 參數。主程序傳給他的是10,(0xa),在memory窗口ebp+8的位置找到他。

            3.?????? 局部變量,我在程序中故意將sum初始化為0xabcd,大家可以在memory窗口ebp-4的位置找到他。


            有興趣地同學可以按F5,在下一個斷點中察看相關信息。

            ?

            好,今天的這片博客中我們對調用棧有了感性的認識,在ebp周圍找到了返回地址,參數以及局部變量。下一片博客中,我將解釋為什么這些信息存儲在這些位置,敬請期待。

            ?

            本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/mountaintaiII/archive/2009/02/08/3869287.aspx

            Feedback

            # re: 小覽call stack(調用棧) (一)   回復  更多評論   

            2010-07-28 13:46 by 好看網
            好東西得支持
            亚洲狠狠婷婷综合久久久久 | 久久婷婷五月综合色高清| 香蕉久久AⅤ一区二区三区| 久久这里有精品| 国产亚洲欧美精品久久久| 久久久精品国产亚洲成人满18免费网站 | 久久综合给久久狠狠97色| 久久国产一区二区| 区亚洲欧美一级久久精品亚洲精品成人网久久久久 | 久久综合狠狠色综合伊人| 色诱久久av| 久久亚洲国产中v天仙www| 亚洲午夜无码久久久久小说| 国产精品久久久久9999高清| 亚洲人成无码www久久久| 久久国产精品99精品国产987| 久久久久久精品久久久久| 99久久免费国产特黄| 精品无码久久久久国产动漫3d| 一本大道久久a久久精品综合| 国产69精品久久久久9999APGF| 国产精品99久久久久久宅男| 久久A级毛片免费观看| 2021最新久久久视精品爱| 久久无码人妻精品一区二区三区 | 久久久久国产成人精品亚洲午夜| 国产亚洲精品久久久久秋霞| 久久久综合香蕉尹人综合网| 久久天天躁狠狠躁夜夜网站| 国内精品伊人久久久影院| 人妻丰满?V无码久久不卡| 国产亚洲精午夜久久久久久| 亚洲国产成人久久精品影视| 99久久免费国产精精品| 国产精品久久国产精麻豆99网站| 国产Av激情久久无码天堂| 久久精品国产亚洲AV电影| 久久久久久久精品成人热色戒| 久久天天婷婷五月俺也去| 一本大道久久东京热无码AV| 久久亚洲精品成人无码网站 |