0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

嵌入式C代碼編碼習(xí)慣要遵循哪些規(guī)則

電子工程師 ? 來源:cc ? 2019-01-21 15:17 ? 次閱讀

Cortex-M這類微控制器編程通常采用C代碼,那么編程人員如何編寫代碼才能讓C編譯器產(chǎn)生高質(zhì)量底層代碼就成為一個(gè)很重要的話題。這里所說的高質(zhì)量底層代碼是指既達(dá)到編程人員意圖又方便編譯器優(yōu)化的代碼。

本文將從編寫利于優(yōu)化的源代碼,節(jié)省棧和內(nèi)存空間,函數(shù)原型,整型和位取反,同時(shí)讀寫變量的保護(hù),不進(jìn)行初始化的變量這幾個(gè)方面來討論如何編寫良好的嵌入式C代碼。

編寫利于優(yōu)化的源代碼

我們在編寫源代碼的時(shí)候如果能夠遵循以下幾點(diǎn),可以讓編譯器更好的對代碼進(jìn)行優(yōu)化:

1)局部變量(自動(dòng)變量和參數(shù))比靜態(tài)或全局變量要更好。為什么這么說呢,因?yàn)閮?yōu)化器會(huì)假定任何一個(gè)函數(shù)都可能修改靜態(tài)或全局變量。當(dāng)局部變量的生命周期結(jié)束的時(shí)候,它所占據(jù)的內(nèi)存就可以被其它變量使用,而全局變量在整個(gè)程序的生命周期內(nèi)都不會(huì)釋放它所占據(jù)的內(nèi)存空間。

2)避免用&運(yùn)算符取局部變量的地址。這里有兩個(gè)原因會(huì)導(dǎo)致該操作的效率低下。首先,變量必須放在內(nèi)存中,不能放在處理器寄存器中,這將導(dǎo)致更長更慢的代碼效率。其次,優(yōu)化器不再假設(shè)其它的函數(shù),因此不會(huì)影響到該變量。

3)編譯器的內(nèi)聯(lián)函數(shù)能力。為了最大限度的影響編譯器的內(nèi)聯(lián)轉(zhuǎn)換,我們最好把那些多個(gè)模塊都用到的小函數(shù)寫在頭文件中而不是實(shí)現(xiàn)文件中。

節(jié)省棧和內(nèi)存空間

以下的編程技術(shù)可以讓我們節(jié)省內(nèi)存和??臻g:

1)如果??臻g有限,那么我們就要盡量避免長的調(diào)用鏈和遞歸函數(shù)。

2)避免使用大的聚合類型(比如結(jié)構(gòu)體)作為參數(shù)或者返回類型。為了節(jié)省棧空間,我們應(yīng)該更多的使用指針來代替這種聚合類型。

函數(shù)原型

有兩種函數(shù)的定義和聲明方式可以使用。一種是原型風(fēng)格,一種是Kernighan & Ritchie C風(fēng)格。兩種風(fēng)格都是可以的,但強(qiáng)烈建議應(yīng)用原型風(fēng)格,也就是說對每一個(gè)公共函數(shù)都在相應(yīng)的頭文件中提供一個(gè)原型聲明。

這是因?yàn)榫幾g器對應(yīng)用Kernighan & Ritchie C風(fēng)格的參數(shù)不進(jìn)行類型檢查。應(yīng)用原型風(fēng)格在某些情況下將產(chǎn)生高效的代碼,因?yàn)樗恍枰M(jìn)行參數(shù)類型提升。為了保證所有的公共函數(shù)都在定義之前聲明過,可以打開編譯器選項(xiàng) Project>Options>C/C++ Compiler>Language 1>Require prototypes

以下是兩種風(fēng)格的示例

1)原型風(fēng)格:

原型風(fēng)格中,必須寫明每個(gè)參數(shù)的類型。

int Test(char, int); /* 聲明 */

int Test(char ch, int i) /* 定義 */

{

return i - ch;

}

2)Kernighan & Ritchie風(fēng)格

Kernighan & Ritchie風(fēng)格中,不需要進(jìn)行函數(shù)原型聲明。取而代之的是一個(gè)空參數(shù)列表的函數(shù)聲明。函數(shù)的定義也有些不同。

int Test(); /* 聲明 */

int Test(ch, i) /* 定義 */

char ch;

int i;

{

return i - ch;

}

整型和位取反

在某些情況下,整數(shù)類型和它們的轉(zhuǎn)換提升規(guī)則會(huì)導(dǎo)致難以理解的行為。這經(jīng)常出現(xiàn)在賦值或者條件表達(dá)式中,這里涉及不同長度類型的數(shù)據(jù)和邏輯操作尤其是位取反操作。這里的類型也包括常數(shù)類型。例如:1個(gè)8位的字符類型,1個(gè)32位的整數(shù)類型,按照二進(jìn)制補(bǔ)碼操作。

void F1(unsigned char c1)

{

if (c1 == ~0x08);

}

這里,測試條件總是false。因?yàn)橛疫叺?x08 = 0x00000008,~0x00000008 = 0xFFFFFFF7。左邊的c1是1個(gè)8位無符號字符類型,因此它不可能比255大,也不可能是負(fù)數(shù),這就意味著它的高24位不可能置1。所以這個(gè)測試條件總是false的。

同時(shí)讀寫變量的保護(hù)

在中斷程序或者單獨(dú)線程中用到的變量經(jīng)常是異步讀寫的,它們必須進(jìn)行適當(dāng)?shù)貥?biāo)記和適當(dāng)?shù)谋Wo(hù)。編譯器應(yīng)用volatile關(guān)鍵字對這類變量進(jìn)行標(biāo)記。這個(gè)關(guān)鍵字通知編譯器該對象的值無任何持久性,不要對它進(jìn)行任何優(yōu)化。

它迫使編譯器每次需要該對象數(shù)據(jù)內(nèi)容的時(shí)候都必須讀該對象,而不是只讀一次數(shù)據(jù)并將它放在處理器的寄存器中以便后續(xù)訪問之用。

不進(jìn)行初始化的變量

通常,運(yùn)行時(shí)環(huán)境在應(yīng)用程序啟動(dòng)的時(shí)候會(huì)初始化所有的靜態(tài)和全局變量。編譯器支持用__no_init關(guān)鍵字來聲明不進(jìn)行初始化的變量。用__no_init關(guān)鍵字聲明的變量通常用在大的數(shù)據(jù)輸入緩沖這樣的地方。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • 嵌入式
    +關(guān)注

    關(guān)注

    5082

    文章

    19126

    瀏覽量

    305198
  • 編碼
    +關(guān)注

    關(guān)注

    6

    文章

    942

    瀏覽量

    54829

原文標(biāo)題:養(yǎng)成良好的嵌入式C代碼編碼習(xí)慣要遵循哪些規(guī)則?

文章出處:【微信號:FANYPCB,微信公眾號:凡億PCB】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    嵌入式開發(fā)習(xí)慣嵌入式gdb調(diào)試步驟

    嵌入式linux開發(fā)過程中,出現(xiàn)BUG是在所難免的,這時(shí)就需要用到調(diào)試技術(shù)了,最常用、最簡單的調(diào)試方法當(dāng)屬printf打印了。 如果ubuntu測試通過,程序下載到板子里出現(xiàn)BUG,串口還出現(xiàn)錯(cuò)誤提示
    發(fā)表于 12-17 23:39

    嵌入式編程要養(yǎng)成好習(xí)慣

    都說字如其人,但作為嵌入式工程師來講,編制程序的習(xí)慣同樣能夠反應(yīng)出其水平,每位工程師都會(huì)有自己的程序設(shè)計(jì)風(fēng)格和邏輯思路,所以想要成為一名合格的嵌入式工程師,形成良好的編程習(xí)慣是非常有必
    發(fā)表于 01-07 16:36

    養(yǎng)成良好的嵌入式C代碼編碼習(xí)慣要遵循哪些規(guī)則?

    養(yǎng)成良好的嵌入式C代碼編碼習(xí)慣要遵循哪些規(guī)則?Cor
    發(fā)表于 04-04 17:14

    嵌入式工程師是如何養(yǎng)成良好的C代碼編碼習(xí)慣?

    方便編譯器優(yōu)化的代碼。本文將從編寫利于優(yōu)化的源代碼,節(jié)省棧和內(nèi)存空間,函數(shù)原型,整型和位取反,同時(shí)讀寫變量的保護(hù),不進(jìn)行初始化的變量這幾個(gè)方面來討論如何編寫良好的嵌入式C
    發(fā)表于 09-04 07:30

    嵌入式Linux C代碼規(guī)范化

    本文檔為作者 在嵌入式嵌入式 linux C語言的學(xué)習(xí)和工作中所總結(jié)的代碼規(guī)范 ,是作者從 STM32單片機(jī)開發(fā)向 Linux C開發(fā)的時(shí)
    發(fā)表于 01-14 10:25

    嵌入式實(shí)時(shí)程序設(shè)計(jì)中C/C++代碼的優(yōu)化

    本文簡單介紹了嵌入式實(shí)時(shí)程序設(shè)計(jì)的特點(diǎn)和嵌入式系統(tǒng)設(shè)計(jì)中語言的選擇,著重介紹了以下幾種在嵌入式實(shí)時(shí)程序設(shè)計(jì)中優(yōu)化 C/C++
    發(fā)表于 08-07 08:47 ?15次下載

    嵌入式C編程

    嵌入式C編程,非常有用的資料,介紹嵌入式C語言編程
    發(fā)表于 12-29 17:29 ?0次下載

    養(yǎng)成良好的嵌入式C代碼編碼習(xí)慣要遵循哪些規(guī)則

    代碼。本文將從編寫利于優(yōu)化的源代碼,節(jié)省棧和內(nèi)存空間,函數(shù)原型,整型和位取反,同時(shí)讀寫變量的保護(hù),不進(jìn)行初始化的變量這幾個(gè)方面來討論如何編寫良好的嵌入式C
    發(fā)表于 01-04 14:34 ?244次閱讀

    優(yōu)化嵌入式軟件時(shí)可以遵循幾個(gè)通用技巧盤點(diǎn)

    早前的專欄中曾討論過在許多情況下需要優(yōu)化的嵌入式系統(tǒng)的關(guān)鍵特征,包括系統(tǒng)時(shí)序、代碼大小、RAM使用率和能耗。雖然優(yōu)化每個(gè)特征通常要求不同的方法和技術(shù),但開發(fā)人員在優(yōu)化嵌入式軟件時(shí)可以遵循
    發(fā)表于 03-08 14:40 ?663次閱讀

    嵌入式代碼之編寫規(guī)范

    嵌入式代碼編碼規(guī)范,用于規(guī)范自己的代碼,增強(qiáng)可讀性,非標(biāo)準(zhǔn)規(guī)范。最好能強(qiáng)制自己形成良好的編碼風(fēng)格,有利于開發(fā)
    的頭像 發(fā)表于 04-26 15:21 ?5280次閱讀

    嵌入式C編碼需要遵循的規(guī)范有哪些?

    編寫代碼時(shí)最重要的一條規(guī)則是:檢查周圍的代碼并嘗試模仿它。
    的頭像 發(fā)表于 06-21 15:07 ?1372次閱讀

    嵌入式外中斷c語言代碼

    嵌入式外中斷c語言代碼(arm嵌入式開發(fā)實(shí)例)-嵌入式外中斷c語言
    發(fā)表于 07-30 11:29 ?4次下載
    <b class='flag-5'>嵌入式</b>外中斷<b class='flag-5'>c</b>語言<b class='flag-5'>代碼</b>

    嵌入式軟件之c語言編碼規(guī)范

    嵌入式軟件之c語言編碼規(guī)范
    發(fā)表于 10-28 18:13 ?28次下載

    嵌入式C++編程

    編程特性來構(gòu)建嵌入式系統(tǒng)您將了解如何將您的系統(tǒng)與外部外圍設(shè)備以及使用驅(qū)動(dòng)程序的有效方式集成指導(dǎo)您測試和優(yōu)化代碼以獲得更好的性能并實(shí)現(xiàn)有用的設(shè)計(jì)模式將了解如何使用 Qt,這是用于構(gòu)建嵌入式系統(tǒng)的流行 GUI 庫。內(nèi)容
    發(fā)表于 11-04 10:36 ?10次下載
    <b class='flag-5'>嵌入式</b><b class='flag-5'>C</b>++編程

    什么是__attribute__?嵌入式C代碼屬性怎么定義?

    嵌入式開發(fā),離不開 C 語言,C語言中有很多語法會(huì)直接或間接影響你代碼的質(zhì)量,下面就來講講__attribute__ 關(guān)鍵字的用法。
    的頭像 發(fā)表于 10-13 15:55 ?6612次閱讀
    什么是__attribute__?<b class='flag-5'>嵌入式</b><b class='flag-5'>C</b><b class='flag-5'>代碼</b>屬性怎么定義?