99精品伊人亚洲|最近国产中文炮友|九草在线视频支援|AV网站大全最新|美女黄片免费观看|国产精品资源视频|精彩无码视频一区|91大神在线后入|伊人终合在线播放|久草综合久久中文

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

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

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

嵌入式開發(fā)中的C語言編程思想

STM32嵌入式開發(fā) ? 來源:STM32嵌入式開發(fā) ? 作者:STM32嵌入式開發(fā) ? 2022-04-08 11:03 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

摘要

本文首先分析了C語言的陷阱和缺陷,對(duì)容易犯錯(cuò)的地方進(jìn)行歸納整理;分析了編譯器語義檢查的不足之處并給出防范措施,以Keil MDK編譯器為例,介紹了該編譯器的特性、對(duì)未定義行為的處理以及一些高級(jí)應(yīng)用;在此基礎(chǔ)上,介紹了防御性編程的概念,提出了編程過程中就應(yīng)該防范于未然的多種措施;提出了測(cè)試對(duì)編寫優(yōu)質(zhì)嵌入式程序的重要作用以及常用測(cè)試方法;最后,本文試圖以更高的層次看待編程,討論一些通用的編程思想。

1 編程風(fēng)格

《計(jì)算機(jī)程序的構(gòu)造和解釋》一書在開篇寫到:程序?qū)懗鰜硎墙o人看的,附帶能在機(jī)器上運(yùn)行。

1.1 整潔的樣式

使用什么樣的編碼樣式一直都頗具爭(zhēng)議性的,比如縮進(jìn)和大括號(hào)的位置。因?yàn)榫幋a的樣式也會(huì)影響程序的可讀性,面對(duì)一個(gè)亂放括號(hào)、對(duì)齊都不一致的源碼,我們很難提起閱讀它的興趣。我們總要看別人的程序,如果彼此編碼樣式相近,讀起源碼來會(huì)覺得比較舒適。但是編碼風(fēng)格的問題是主觀的,永遠(yuǎn)不可能在編碼風(fēng)格上達(dá)成統(tǒng)一意見。因此只要你的編碼樣式整潔、結(jié)構(gòu)清晰就足夠了。除此之外,對(duì)編碼樣式再?zèng)]有其它要求。

提出匈牙利命名法的程序員、前微軟首席架構(gòu)師Charles Simonyi說:我覺得代碼清單帶給人的愉快同整潔的家差不多。你一眼就能分辨出家里是雜亂無章還是整潔如新。這也許意義不大。因?yàn)楣馐欠孔诱麧嵳f明不了什么,它仍可能藏污納垢!但是第一印象很重要,它至少反映了程序的某些方面。我敢打賭,我在3米開外就能看出程序拙劣與否。我也許沒法保證它很不錯(cuò),但如果從3米外看起來就很糟,我敢保證這程序?qū)懙貌挥眯?。如果寫得不用心,那它在邏輯上也許就不會(huì)優(yōu)美。

1.2 清晰的命名

變量、函數(shù)、宏等等都需要命名,清晰的命名是優(yōu)秀代碼的特點(diǎn)之一。命名的要點(diǎn)之一是名稱應(yīng)能清晰的描述這個(gè)對(duì)象,以至于一個(gè)初級(jí)程序員也能不費(fèi)力的讀懂你的代碼邏輯。我們寫的代碼主要給誰看是需要思考的:給自己、給編譯器還是給別人看?我覺得代碼最主要的是給別人看,其次是給自己看。如果沒有一個(gè)清晰的命名,別人在維護(hù)你的程序時(shí)很難在整個(gè)全貌上看清代碼,因?yàn)橐涀∈鄠€(gè)以上的糟糕命名的變量是件非常困難的事;而且一段時(shí)間之后你回過頭來看自己的代碼,很有可能不記得那些糟糕命名的變量是什么意思。

為對(duì)象起一個(gè)清晰的名字并不是簡(jiǎn)單的事情。首先能認(rèn)識(shí)到名稱的重要性需要有一個(gè)過程,這也許跟譚式C程序教材被大學(xué)廣泛使用有關(guān):滿書的a、b、c、x、y、z變量名是很難在關(guān)鍵的初學(xué)階段給人傳達(dá)優(yōu)秀編程思想的;其次如何恰當(dāng)?shù)臑閷?duì)象命名也很有挑戰(zhàn)性,要準(zhǔn)確、無歧義、不羅嗦,要對(duì)英文有一定水平,所有這些都要滿足時(shí),就會(huì)變得很困難;此外,命名還需要考慮整體一致性,在同一個(gè)項(xiàng)目中要有統(tǒng)一的風(fēng)格,堅(jiān)持這種風(fēng)格也并不容易。

關(guān)于如何命名,Charles Simonyi說:面對(duì)一個(gè)具備某些屬性的結(jié)構(gòu),不要隨隨便便地取個(gè)名字,然后讓所有人去琢磨名字和屬性之間有什么關(guān)聯(lián),你應(yīng)該把屬性本身,用作結(jié)構(gòu)的名字。

1.3 恰當(dāng)?shù)淖⑨?/p>

注釋向來也是爭(zhēng)議之一,不加注釋和過多的注釋我都是反對(duì)的。不加注釋的代碼顯然是很糟糕的,但過多的注釋也會(huì)妨礙程序的可讀性,由于注釋可能存在的歧義,有可能會(huì)誤解程序真實(shí)意圖,此外,過多的注釋會(huì)增加程序員不必要的時(shí)間。如果你的編碼樣式整潔、命名又很清晰,那么,你的代碼可讀性不會(huì)差到哪去,而注釋的本意就是為了便于理解程序。

這里建議使用良好的編碼樣式和清晰的命名來減少注釋,對(duì)模塊、函數(shù)、變量、數(shù)據(jù)結(jié)構(gòu)、算法和關(guān)鍵代碼做注釋,應(yīng)重視注釋的質(zhì)量而不是數(shù)量。如果你需要一大段注釋才能說清楚程序做什么,那么你應(yīng)該注意了:是否是因?yàn)槌绦蜃兞棵粔蚯逦蛘叽a邏輯過于混亂,這個(gè)時(shí)候你應(yīng)該考慮的可能就不是注釋,而是如何精簡(jiǎn)這個(gè)程序了。

2 數(shù)據(jù)結(jié)構(gòu)

數(shù)據(jù)結(jié)構(gòu)是程序設(shè)計(jì)的基礎(chǔ)。在設(shè)計(jì)程序之前,應(yīng)該先考慮好所需要的數(shù)據(jù)結(jié)構(gòu)。

前微軟首席架構(gòu)師Charles Simonyi:編程的第一步是想象。就是要在腦海中對(duì)來龍去脈有極為清晰的把握。在這個(gè)初始階段,我會(huì)使用紙和鉛筆。我只是信手涂鴉,并不寫代碼。我也許會(huì)畫些方框或箭頭,但基本上只是涂鴉,因?yàn)檎嬲南敕ㄔ谖夷X海里。我喜歡想象那些有待維護(hù)的結(jié)構(gòu),那些結(jié)構(gòu)代表著我想編碼的真實(shí)世界。一旦這個(gè)結(jié)構(gòu)考慮得相當(dāng)嚴(yán)謹(jǐn)和明確,我便開始寫代碼。我會(huì)坐到終端前,或者換在以前的話,就會(huì)拿張白紙,開始寫代碼。這相當(dāng)容易。我只要把頭腦中的想法變換成代碼寫下來,我知道結(jié)果應(yīng)該是什么樣的。大部分代碼會(huì)水到渠成,不過我維護(hù)的那些數(shù)據(jù)結(jié)構(gòu)才是關(guān)鍵。我會(huì)先想好數(shù)據(jù)結(jié)構(gòu),并在整個(gè)編碼過程中將它們牢記于心。

開發(fā)過以太網(wǎng)操作系統(tǒng)SDS 940的Butler Lampson:(程序員)最重要的素質(zhì)是能夠把問題的解決方案組織成容易操控的結(jié)構(gòu)。

開發(fā)CP/M操作系統(tǒng)的Gary.A:如果不能確認(rèn)數(shù)據(jù)結(jié)構(gòu)是正確的,我是決不會(huì)開始編碼的。我會(huì)先畫數(shù)據(jù)結(jié)構(gòu),然后花很長時(shí)間思考數(shù)據(jù)結(jié)構(gòu)。在確定數(shù)據(jù)結(jié)構(gòu)之后我就開始寫一些小段的代碼,并不斷地改善和監(jiān)測(cè)。在編碼過程中進(jìn)行測(cè)試可以確保所做的修改是局部的,并且如果有什么問題的話,能夠馬上發(fā)現(xiàn)。

微軟創(chuàng)始人比爾**·**蓋茨:編寫程序最重要的部分是設(shè)計(jì)數(shù)據(jù)結(jié)構(gòu)。接下來重要的部分是分解各種代碼塊。

編寫世界上第一個(gè)電子表格軟件的Dan Bricklin:在我看來,寫程序最重要的部分是設(shè)計(jì)數(shù)據(jù)結(jié)構(gòu),此外,你還必須知道人機(jī)界面會(huì)是什么樣的。

我們舉個(gè)例子來說明。在介紹防御性編程的時(shí)候,提到公司使用的LCD顯示屏抗干擾能力一般,為了提高LCD的穩(wěn)定性,需要定期讀出LCD內(nèi)部的關(guān)鍵寄存器值,然后跟存在Flash中的初始值相比較。需要讀出的LCD寄存器有十多個(gè),從每個(gè)寄存器讀出的值也不盡相同,從1個(gè)到8個(gè)字節(jié)都有可能。如果不考慮數(shù)據(jù)結(jié)構(gòu),編寫出的程序?qū)?huì)很冗長。

void lcd_redu(void)

{ 。

讀第一個(gè)寄存器值;

if(第一個(gè)寄存器值==Flash存儲(chǔ)值)

{

讀第二個(gè)寄存器值;

if(第二個(gè)寄存器值==Flash存儲(chǔ)值)

{

。..

讀第十個(gè)寄存器值;

if(第十個(gè)寄存器值==Flash存儲(chǔ)值)

{

返回;

}

else

{

重新初始化LCD;

}

}

else

{

重新初始化LCD;

}

}

else

{

重新初始化LCD;

}

}

我們分析這個(gè)過程,發(fā)現(xiàn)能提取出很多相同的元素,比如每次讀LCD寄存器都需要該寄存器的命令號(hào),都會(huì)經(jīng)過讀寄存器、判斷值是否相同、處理異常情況這一過程。所以我們可以提取一些相同的元素,組織成數(shù)據(jù)結(jié)構(gòu),用統(tǒng)一的方法去處理這些數(shù)據(jù),將數(shù)據(jù)與處理過程分開來。

我們可以先提取相同的元素,將之組織成數(shù)據(jù)結(jié)構(gòu):

c2ecb04c-b659-11ec-aa7f-dac502259ad0.png

這里lcd_command表示的是LCD寄存器命令號(hào);lcd_get_value是一個(gè)數(shù)組,表示寄存器要初始化的值,這是因?yàn)閷?duì)于一個(gè)LCD寄存器,可能要初始化多個(gè)字節(jié),這是硬件特性決定的;lcd_value_num是指一個(gè)寄存器要多少個(gè)字節(jié)的初值,這是因?yàn)槊恳粋€(gè)寄存器的初值數(shù)目是不同的,我們用同一個(gè)方法處理數(shù)據(jù)時(shí),是需要這個(gè)信息的。

就本例而言,我們將要處理的數(shù)據(jù)都是事先固定的,所以定義好數(shù)據(jù)結(jié)構(gòu)后,我們可以將這些數(shù)據(jù)組織成表格:

/*LCD部分寄存器設(shè)置值列表*/

lcd_redu_list_struct const lcd_redu_list_str[]= {

{SSD1963_Get_Address_Mode,{0x20}

,1}, /*1*/

{SSD1963_Get_Pll_Mn

,{0x3b,0x02,0x04}

,3}, /*2*/

{SSD1963_Get_Pll_Status

,{0x04}

,1}, /*3*

{SSD1963_Get_Lcd_Mode

,{0x24,0x20,0x01,0xdf,0x01,0x0f,0x00}

,7}, /*4*/

{SSD1963_Get_Hori_Period ,{0x02,0x0c,0x00,0x2a,0x07,0x00,0x00,0x00},8}, /*5*/

{SSD1963_Get_Vert_Period ,{0x01,0x1d,0x00,0x0b,0x09,0x00,0x00}

,7}, /*6*/ {SSD1963_Get_Power_Mode ,{0x1c}

,1}, /*7*/ {SSD1963_Get_Display_Mode,{0x03}

,1}, /*8*/ {SSD1963_Get_Gpio_Conf ,{0x0F,0x01}

,2}, /*9*/ {SSD1963_Get_Lshift_Freq ,{0x00,0xb8}

,2}, /*10* };

至此,我們就可以用一個(gè)處理過程來完成數(shù)十個(gè)LCD寄存器的讀取、判斷和異常處理了:

/** * lcd 顯示冗余

* 每隔一段時(shí)間調(diào)用該程序一次 */ void lcd_redu(void) {

uint8_t tmp[8];

uint32_t i,j;

uint32_t lcd_init_flag;

lcd_init_flag =0;

for(i=0;i《sizeof(lcd_redu_list_str)/sizeof(lcd_redu_list_str[0]);i++)

{

LCD_SendCommand(lcd_redu_list_str[i].lcd_command);

uyDelay(10);

for(j=0;j《lcd_redu_list_str[i].lcd_value_num;j++)

{

tmp[j]=LCD_ReadData();

if(tmp[j]!=lcd_redu_list_str[i].lcd_get_value[j])

{

lcd_init_flag=0x55;

//一些調(diào)試語句,打印出錯(cuò)的具體信息

goto handle_lcd_init;

}

}

}

handle_lcd_init:

if(lcd_init_flag==0x55)

{

//重新初始化LCD

//一些必要的恢復(fù)措施

}

}

通過合理的數(shù)據(jù)結(jié)構(gòu),我們可以將數(shù)據(jù)和處理過程分開,LCD冗余判斷過程可以用很簡(jiǎn)潔的代碼來實(shí)現(xiàn)。更重要的是,將數(shù)據(jù)和處理過程分開更有利于代碼的維護(hù)。比如,通過實(shí)驗(yàn)發(fā)現(xiàn),我們還需要增加一個(gè)LCD寄存器的值進(jìn)行判斷,這時(shí)候只需要將新增加的寄存器信息按照數(shù)據(jù)結(jié)構(gòu)格式,放到LCD寄存器設(shè)置值列表中的任意位置即可,不用增加任何處理代碼即可實(shí)現(xiàn)!這僅僅是數(shù)據(jù)結(jié)構(gòu)的優(yōu)勢(shì)之一,使用數(shù)據(jù)結(jié)構(gòu)還能簡(jiǎn)化編程,使復(fù)雜過程變的簡(jiǎn)單,這個(gè)只有實(shí)際編程后才會(huì)有更深的理解。

審核編輯 :李倩

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

    關(guān)注

    180

    文章

    7632

    瀏覽量

    141681
  • 嵌入式開發(fā)
    +關(guān)注

    關(guān)注

    18

    文章

    1075

    瀏覽量

    48862

原文標(biāo)題:嵌入式開發(fā)中的C語言編程思想

文章出處:【微信號(hào):c-stm32,微信公眾號(hào):STM32嵌入式開發(fā)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    嵌入式開發(fā)就業(yè)還有前景嗎?

    的電路設(shè)計(jì)和編程能力。 2)醫(yī)療設(shè)備 醫(yī)療設(shè)備是指用于診斷、治療和輔助治療的各種設(shè)備。隨著醫(yī)療技術(shù)的不斷發(fā)展,醫(yī)療設(shè)備市場(chǎng)也在不斷擴(kuò)大,嵌入式開發(fā)在醫(yī)療設(shè)備領(lǐng)域的應(yīng)用前景也十分廣闊。在醫(yī)療設(shè)備領(lǐng)域
    發(fā)表于 07-16 10:47

    嵌入式開發(fā)入門指南:從零開始學(xué)習(xí)嵌入式

    特定功能的計(jì)算機(jī)系統(tǒng),廣泛應(yīng)用于智能家居、工業(yè)控制、醫(yī)療設(shè)備、車載系統(tǒng)等領(lǐng)域。 2. 學(xué)習(xí)嵌入式開發(fā)的前置知識(shí)熟悉C語言編程掌握基本的數(shù)據(jù)結(jié)構(gòu)與算法了解數(shù)字電路與微控制器原理熟悉Lin
    發(fā)表于 05-15 09:29

    嵌入式開發(fā):高門檻的系統(tǒng)性工程與 996 的行業(yè)困局

    嵌入式開發(fā)的門檻,往往被培訓(xùn)機(jī)構(gòu)和表象所掩蓋。許多人誤以為 “用 C 語言寫個(gè)跑在 ARM 上的程序” 就是嵌入式,實(shí)則連皮毛都未觸及。真正的嵌入式
    的頭像 發(fā)表于 04-09 11:06 ?339次閱讀
    <b class='flag-5'>嵌入式開發(fā)</b>:高門檻的系統(tǒng)性工程與 996 的行業(yè)困局

    Python在嵌入式系統(tǒng)的應(yīng)用場(chǎng)景

    你想把你的職業(yè)生涯提升到一個(gè)新的水平?Python在嵌入式系統(tǒng)中正在成為一股不可缺少的新力量。盡管傳統(tǒng)上嵌入式開發(fā)更多地依賴于CC++語言
    的頭像 發(fā)表于 03-19 14:10 ?707次閱讀

    AI來襲!嵌入式開發(fā)者該如何應(yīng)對(duì)轉(zhuǎn)型?

    設(shè)計(jì)和低功耗的編程方式,但現(xiàn)在,AI技術(shù)的涌入要求我們?nèi)绾螒?yīng)對(duì)轉(zhuǎn)型,如何將傳統(tǒng)的嵌入式開發(fā)與新興的AI技術(shù)融合,是每個(gè)嵌入式開發(fā)者必須思考的問題。今天,我們就來聊聊
    的頭像 發(fā)表于 03-04 14:41 ?818次閱讀
    AI來襲!<b class='flag-5'>嵌入式開發(fā)</b>者該如何應(yīng)對(duì)轉(zhuǎn)型?

    如何成為嵌入式開發(fā)工程師?

    和使用方法。- 掌握接口標(biāo)準(zhǔn)和總線協(xié)議,如UART、SPI、I2C、CAN等。 3. 編程能力:- 精通至少一種編程語言,尤其是C/
    發(fā)表于 02-19 10:39

    新手怎么學(xué)嵌入式?

    的運(yùn)行機(jī)制。例如,了解數(shù)據(jù)結(jié)構(gòu)的鏈表、棧和隊(duì)列,對(duì)于在嵌入式編程管理數(shù)據(jù)非常有幫助。 2. 選擇合適的編程
    發(fā)表于 12-12 10:51

    如何使用 RISC-V 進(jìn)行嵌入式開發(fā)

    RISC-V是一種開源的指令集架構(gòu)(ISA),它允許任何人設(shè)計(jì)、制造和銷售基于RISC-V的處理器,這為嵌入式開發(fā)提供了極大的靈活性和創(chuàng)新空間。以下是使用RISC-V進(jìn)行嵌入式開發(fā)的基本步驟: 一
    的頭像 發(fā)表于 12-11 17:32 ?1937次閱讀

    一文了解嵌入式軟件開發(fā)的對(duì)象

    以前應(yīng)用場(chǎng)景很單一,嵌入式開發(fā)可能談不上面向?qū)ο?b class='flag-5'>開發(fā)。但現(xiàn)在,做嵌入式開發(fā),沒有面向?qū)ο?b class='flag-5'>開發(fā),你就有點(diǎn)落伍了。 本文結(jié)合個(gè)人經(jīng)驗(yàn)和周立功《抽象接口技術(shù)和組件
    的頭像 發(fā)表于 11-15 10:31 ?844次閱讀
    一文了解<b class='flag-5'>嵌入式</b>軟件<b class='flag-5'>開發(fā)</b>的對(duì)象

    MCU編程語言開發(fā)環(huán)境介紹

    微控制器單元(Microcontroller Unit,簡(jiǎn)稱MCU)是嵌入式系統(tǒng)的核心,廣泛應(yīng)用于各種電子產(chǎn)品。隨著技術(shù)的發(fā)展,MCU編程語言
    的頭像 發(fā)表于 11-01 11:51 ?2194次閱讀

    零基礎(chǔ)嵌入式開發(fā)學(xué)習(xí)路線

    嵌入式開發(fā)”沒有接觸過的同學(xué)可能會(huì)不明覺厲,但是只要你了解了,感興趣并且有一個(gè)正確的學(xué)習(xí)路線的話,零基礎(chǔ)也能入門。給大家介紹一個(gè)簡(jiǎn)單易懂的學(xué)習(xí)路線,讓你能夠從零開始學(xué)習(xí)嵌入式開發(fā)。 第一步:掌握
    發(fā)表于 10-25 15:55

    嵌入式開發(fā)常見問題排查

    嵌入式開發(fā)問題排查很多人認(rèn)為嵌入式開發(fā)很難,主要是因?yàn)樵谶@個(gè)過程中常常會(huì)遇到各式各樣的問題。這些問題的復(fù)雜性和多樣性使得許多人感到困惑和無所適從。然而,如果將這些問題逐一拆解,實(shí)際上大部分都可以
    的頭像 發(fā)表于 09-22 08:04 ?827次閱讀
    <b class='flag-5'>嵌入式開發(fā)</b>常見問題排查

    嵌入式開發(fā)常用軟件有哪些?

    嵌入式開發(fā)常用軟件有哪些? 對(duì)于初學(xué)嵌入式的朋友們,會(huì)想要了解嵌入式軟件開發(fā)常用的軟件有些,有什么用。那么看以下常用的軟件介紹。 1.VisualStudio Code 簡(jiǎn)稱VSCod
    發(fā)表于 09-09 15:22

    AWFlow:內(nèi)置豐富的功能節(jié)點(diǎn),簡(jiǎn)化嵌入式開發(fā)流程

    AWFlow突破傳統(tǒng)C語言限制的嵌入式開發(fā)框架,通過封裝硬件驅(qū)動(dòng)和系統(tǒng)平臺(tái),集成通用軟件功能與通信協(xié)議,助力開發(fā)者專注于核心業(yè)務(wù),提升效率,縮短周期,迅速適應(yīng)市場(chǎng)變化。在傳統(tǒng)的
    的頭像 發(fā)表于 08-09 08:25 ?531次閱讀
    AWFlow:內(nèi)置豐富的功能節(jié)點(diǎn),簡(jiǎn)化<b class='flag-5'>嵌入式開發(fā)</b>流程

    聚焦嵌入式開發(fā)的合規(guī)性工具、項(xiàng)目管理工具、版本迭代工具應(yīng)用

    ,就嵌入式開發(fā)與管理領(lǐng)域的最新趨勢(shì)、工具選擇以及DevSecOps實(shí)踐應(yīng)用等方面展開了深入探討。 本期對(duì)話龍智資深DevSecOps顧問徐晨暉, 分享嵌入式開發(fā)的合規(guī)性工具、項(xiàng)目管理和版本管理的工具選擇,以及DevSecOps
    的頭像 發(fā)表于 07-29 15:15 ?1082次閱讀