引言 計算機編程語言的關鍵字就好比是它的靈魂,只有深入理解了它們的含義才能編寫出優秀的代碼。C語言以其簡潔、高效和強大等特性成為嵌入式軟件編程的首選語言,但是某些關鍵字,例如const、static、extern和volatile等,在不同的場合具有不同的含義,而且某些用法晦澀難懂,為此本文詳細介紹這些關鍵字的用法及其背后的原理。 1 const const限定的對象表示編譯器可以將它放在只讀存儲器中,也就意味著在對其進行初始化之后就不能改變它的值。根據const使用的不同場合,大致可以分為三種情況,其一限定普通變量,其二限定函數參數,其三限定指針變量。 第一和第二種情況最為簡單,語句①和語句②分別展示了它的用法。語句①定義了一個值為10的整型常量。語句②中的const表示在函數體中不能修改src 指向的區域中的數據,這與函數的拷貝功能相對應,只做它應該做的事情而不應該有其他副作用,編譯器可以利用這些信息進行適當的優化。 ①const int i=10; ②void*memcpy(void * dst,const void * src,size_t size); ③const int *ptr; ④int const *ptr; ⑤int*const ptr; ⑥int const*cons ptr; 第3種情況最為復雜,雖然只是const位置不同,但是卻可能具有完全不同的意義。一般,一個聲明語句由聲明說明符(decl-specifier)和一系列聲明子(declarator)兩部分組成,而且聲明說明符中的符號可以以任何次序出現。理解聲明的第一步是定位說明符和聲明子的邊界。這很容易:所有的說明符都是關鍵字或者類型名,因此說明符終止于第一個不是以上類型之一的符號。例如,在語句③和④中第一個既不是關鍵字也不是類型名的符號是“*”,即聲明說明符分別為const.int和int const,由于聲明說明符中的符號可以以任意次序出現,因此語句③和④的含義是相同的。 為了迅速弄清語句表達的含義,參考文獻[1]介紹了一種簡便的方法,其要點就是“逆序讀出定義”,如圖1所示。 ![]() 2 static與extem static的含義隨著出現位置(全局變量還是局部變量)和修飾對象(變量還是函數)的不同而有很大的差別。下面各條目中的模塊指的是一個源文件或者一個翻譯單元: ①位于函數體中的靜態變量在多次函數調用間會維持其值。 ②位于模塊內(但在函數體外)的靜態變量可以被模塊內的所有函數訪問,但不能被模塊外其他函數訪問。也就是說,它是一個本地的全局變量。 ③位于模塊內的靜態函數只能被此模塊內的其他函數調用。也就是說,這個函數的作用域為聲明所在的模塊。 ![]() 為了清楚地理解static的3種用法,必須首先了解C語言中每個標識符都具有的作用域、鏈接和存儲持續期等特性的含義。在ISO C99標準中,其定義如下: ①對象的作用域指的是它僅在程序的某個區域中是可見的(即可以使用)。常見的作用域有文件作用域和塊作用域。 ②對象的存儲持續期決定對象的生命周期,即在程序執行某段區間中為對象保留存儲區。有兩種類型的存儲持續期:靜態的和自動的。靜態存儲持續期的對象的生命周期為程序執行的全過程,它的值在程序啟動前僅初始化一次。 ③鏈接指的是在不同作用域中聲明的或者同一個作用域中多次聲明的標識符可以引用相同的對象或函數。有3種類型的鏈接:外部、內部和無。在情況②和③ 中,static分別用來修飾全局變量glob-al和函數foo,改變它們的鏈接特性,使它們具有內部鏈接。也就是說,只有在定義它們的翻譯單元或者文件內才能使用它們,這對于創建模塊化的軟件非常重要。 與static相反,extern修飾的對象或函數具有外部鏈接。對于那些暴露給外部使用的接口函數應該使用ex-tern限定,那些非接口函數,例如工具函數或與實現細節相關的函數,則應該顯式地使用static限定。這是因為如果函數聲明不帶任何存儲類說明符,那么它具有外部鏈接就好像使用了 extern一樣。 在情況①中,static用來修飾局部變量local,將local的存儲持續期由自動的改變為靜態的,這樣在foo函數的多次調用間會為其保留值。注意作用域、鏈接和存儲持續期特性之間是正交的。例如在情況①中,雖然變量local的存儲持續期變成靜態的,但是它的作用域仍然是塊作用域。 3 volatile volatile關鍵字用來聲明這樣的對象,它們的值可能由于程序控制之外的事件而被潛在改變。volatile強制編譯器不會對其所限定的對象進行任何優化,每次讀寫都必須訪問實際的存儲器而不能使用寄存器中的副本。在實踐中,它大量的用來描述一個對應于內存映射的輸入/輸出端口,例如飛利浦公司 LPC21xx系列ARM處理器的向量地址寄存器定義為: #define VICVectAddr (*((volatile unsigned long*)0xFFFFF030)) 其次,中斷服務例程中使用的非自動變量或者多線程應用程序中多個任務共享的變量也必須使用volatile進行限定。例如在下面的示例中,如果沒有使用 volatile限定g_Flag變量,編譯器看到在foo函數中并沒有修改g_Flag,可能只執行一次g_Flag讀操作并將g_Flag的值緩存在寄存器中,以后每次g_Flag讀操作都使用寄存器中的緩存值而不進行存儲器訪問,導致some_action函數永遠無法執行。 ![]() 4 Dacked 在嵌入式軟件編程中,經常需要精確控制結構體在內存中的布局和訪問非自然對齊的數據,但是C語言標準中并沒有統一的規定而是留給編譯器廠商自行處理。在 ARM C編譯器中,使用__packed關鍵字將任何類型的對齊設置為1字節。在實踐中,__packed主要有兩個功能:其一,當它修飾指針時,表示此指針指向的地址是非自然對齊的,編譯器會生成特殊的代碼以確保獲得正確的結果;其二,當它修飾結構體、聯合或它們中的域時,可以用來創建沒有填充的結構。 與其他RISC架構一樣,ARM處理器能夠高效地訪問對齊的數據,即字地址的末尾兩位為零,半字地址的最后一位為零,也稱這樣的數據位于它的自然大小邊界或者是自然對齊的。ARM編譯器希望普通的“C”指針指向一個4字節對齊內存地址,這樣它可以在代碼中使用LDR/STR指令一次操作4個字節,否則只能使用LDRB/LDRH等字節/半字操作指令。相反如果指針指向一個非自然對齊的地址,例如如果一個整型指針指向地址0x8006,當然希望裝載地址 0xS006-0xS007-0x8008-0xS009處的數據,但是實際上ARM會對非自然對齊的地址進行轉換而從裝載地址 0xS004-0xS005-0x8006-0xS007處的數據。在下面的示例中(測試環境為uVision3),首先定義了一個大小為16字節的整型數組,依次初始化為0,1,2,…,15。由于array是一個整型數組,編譯器會確保它是4字節對齊的,即指針pc指向一個4字節對齊的地址。運行程序后,可以看到如果對pc指針不加__packed標記進行修飾,將得到一個奇怪的0x01000302;而在添加了__packed關鍵字之后,就得到了正確的結果。也就是說,如果要訪問非自然對齊的數據,必須使用__packed關鍵字顯式地標記出來。 ![]() ARM編譯器總是保證程序中的變量、結構體或聯合中的域分配到自然對齊的地址。這意味著編譯器經常需要在各個域之間插入填充,以確保每個域的自然對齊。通常來說,程序員可以對這些填充視而不見,但是也有例外,例如為了節省結構體占用的空間,可以利用__packed去除填充。在了解了編譯器的填充行為之后,可以通過調整域的順序來減小結構體占用的空間。例如雖然結構體s1和s2的域相同,但是sizeof(s1)等于16,而sizeof(s2)等于 12。 參考文獻 1. ARM Ltd.Use of "const" and "volatile". 2. Nigel Jones.A "C" Test:The 0x10 Best Questions for Would-be Embedded Programmers.Embedded System Programming[J/OL],2000(5)[2009-05].http://www.embedded.com/2000/0005/0005feat2.htm. 3. ISO/IEC 9899 WGl4/N1124[OL].[2009-05].http://www.open-std.org/JTC1/SC22/wg14/www/does/nll24.pdf. 作者:青島職業技術學院 劉浩 來源:單片機與嵌入式系統 2009(10) |