通常我們都是學了標準c語言教程后從事單片機c語言的編寫的, 那就先要明白一點, 標準c語言實際上是起源于pc平臺上的一種語言, 標準c語言肯定是不會照顧到單片機的特殊性的. 因此單片機c編譯器中的c語言是一種基于標準c,但是又有相應修改擴充的擴展c語言.所以在單片機c編譯器里寫程序時一定要了解單片機編譯器擴展c語言的不同之處, 絕不能死板地照搬標準c.在標準c里, 局部變量是函數(shù)在調(diào)用的時候才臨時分配存儲空間的,全局變量是程序整個生命周期都一直存在的.不過要知道,臨時分配存儲空間是需要操作系統(tǒng)內(nèi)存管理程序支持的, 單片機中通常都沒有操作系統(tǒng),也就不能實現(xiàn)像pc平臺中那樣的局部變量的空間分配.這里就需要深入了解一下單片機的c編譯器究竟是如何處理局部變量的,如果對此沒有概念,碰到調(diào)試過程中的一些奇異現(xiàn)象恐怕只能覺匪夷所思了.
本文引用地址:http://butianyuan.cn/article/201611/322315.htm另外需要知道的一點是, 不同的編譯器對于局部變量的處理方法也不一樣, 不能學了一個就到處照搬. 這里拿KEIL C ,IARAVR, ICCAVR這個三個編譯器做分析比較.
首先說 Keil C51 , 它的局部變量并不是在堆棧中, C51為了提高代碼的效率, 根據(jù) 51 處理器的特性. 編譯器對函數(shù)局部變量的安排進行了處理.局部變量如果不能分配到 寄存器里, 就放在 RAM 中了.編譯器通過覆蓋分析, 可以共享局部變量的地址空間.。最終的DATA使用量取決于調(diào)用鏈中那個使用DATA最多的鏈。所以,在程序中增加一個局部變量,如果不是位于那個使用DATA最多的鏈中,需要的DATA數(shù)量是有可能不會增加的。
如:main()->f11()->f12()->f13()....//鏈1
|----->f21()->f22()->f23()....//鏈2
因為f11(),f21()不在同一個調(diào)用鏈上,顯然,f11()中使用的局部變量,可以和f21()中的局部變量,使用同一個存儲單元。因為它們中的任何一個處在生命期內(nèi)的話,另一個必然已經(jīng)離開它的生命周期,同時它的局部變量也離開了它的生命周期,這些局部變量所占用的存儲單元當然可以另做它用了。
假設鏈1目前的局部變量需要50個存儲單元,鏈2需要40個存儲單元。那么你在鏈2中加入不多于10個單元的局部變量的話,程序最終需要的存儲單元數(shù)量是不會增加的。
再說ICCAVR ,它把局部變量存放在軟件堆??臻g中.ICCAVR使用兩個堆棧:一個用于子程序調(diào)用和中斷操作的硬件堆棧,一個用于傳遞參數(shù)、臨時變量和局部變量的軟件堆棧。硬件堆棧是從數(shù)據(jù)內(nèi)存的頂部開始分配的,在硬件堆棧下面再分配一定數(shù)量的字節(jié)作為軟件堆棧。
IARAVR對于局部變量的處理方法與ICCAVR一樣. 它也有兩個堆棧,一個是data stack ,一個是return address stack.分別用于存放臨時變量,局部變量,傳遞參數(shù), 和函數(shù)返回地址.
這里需要注意的是, 局部變量存放在堆棧中的處理方式一定要保證堆棧足夠大, 特別是定義了局部數(shù)組變量的情況下,一旦數(shù)組過大,超過了堆棧大小就會發(fā)生堆棧溢出,如果只是讀取數(shù)據(jù)還好, 一旦寫入數(shù)據(jù),就會破壞堆??臻g以外的數(shù)據(jù), 導致程序時常.
評論