今天看到大佬肖遙兄分享的一篇文章:【架構(gòu)篇】嵌入式編程中如何給代碼的結(jié)構(gòu)分層
提到了高內(nèi)聚,低耦合,軟件分層等等的概念。之前又有小伙伴在后臺留言說讓我分享一篇這樣的文章,所以今天它來了!
廢話不多說,理論講太多沒啥感覺,這些條條框框本質(zhì)就是基于面對對象的設(shè)計模式相關(guān)的一些理論,設(shè)計模式就是前人實踐多了發(fā)現(xiàn)一些規(guī)律然后總結(jié)出來的那么一套好用的框架,所以咱們直接出干貨,硬肝!以小熊派上的SPI OLED驅(qū)動為例,將原來開發(fā)包里的LCD驅(qū)動做一些簡單的改造,然后我們根據(jù)需求設(shè)計如下的驅(qū)動模型框架,分為模型、驅(qū)動、設(shè)備三個部分,我們先不考慮太細節(jié)的東西,也不會把這個東西一開始就做得特別復雜,這樣不利于理解,于是我們構(gòu)建如下的框架思維導圖:
1、LCD驅(qū)動框架數(shù)據(jù)結(jié)構(gòu)
框架提供一些什么能力呢?我是這么來做的,非常簡單:
這里提供了將驅(qū)動框架與驅(qū)動進行對接的能力,&lcd_driver
拿到的是定義在驅(qū)動文件里的一個已經(jīng)賦值了的結(jié)構(gòu)體lcd_driver
,這樣,當我在別的地方定義一個LCD_Driver_Model
的變量,就可以將這個變量與驅(qū)動結(jié)構(gòu)體進行對接,這樣就可以通過這個變量來操作驅(qū)動結(jié)構(gòu)體里的接口了。
2、LCD驅(qū)動數(shù)據(jù)結(jié)構(gòu)
LCD驅(qū)動這個數(shù)據(jù)結(jié)構(gòu)要做的事情就是提供操作LCD驅(qū)動能力的接口,這個接口的設(shè)計與硬件無關(guān)。
前面1小節(jié)說過,驅(qū)動框架依賴于驅(qū)動接口,這樣的話我們需要實現(xiàn)驅(qū)動接口里的方法,在對應的方法里,我們要去調(diào)用LCD設(shè)備相關(guān)的接口,進而去操作LCD設(shè)備,以下是接口對應的實現(xiàn):
3、LCD設(shè)備數(shù)據(jù)結(jié)構(gòu)
LCD設(shè)備所需要做的事情就是將這個數(shù)據(jù)結(jié)構(gòu)里的功能函數(shù)與真實的LCD驅(qū)動接口進行對接。比如我們看LCD_Init
這個接口的實現(xiàn),這個就是真實調(diào)用LCD的真實硬件操作了:
4、使用方法
int?main(void)
{
????/*?USER?CODE?BEGIN?1?*/
????LCD_Driver_Model??lcd_model?;
??LCD_Ascii_Show_Para?ascii_para[]?=
????{
????????{80,?100,?240-80,?"RED",?BLACK,?RED,?32},
????????{80,?100,?240-80,?"GREEN",?BLACK,?GREEN,?32},
????????{80,?100,?240-80,?"BLUE",?BLACK,?BLUE,?32},
????};
??LCD_Fill_Para?fill_para[]?=?
??{
???{ascii_para[0].x,ascii_para[0].max_width,ascii_para[0].y,ascii_para[0].y+32},
???{ascii_para[1].x,ascii_para[1].max_width,ascii_para[1].y,ascii_para[1].y+32},
???{ascii_para[2].x,ascii_para[2].max_width,ascii_para[2].y,ascii_para[2].y+32},
??}?;
????/*?USER?CODE?END?1?*/
????/*?MCU?Configuration--------------------------------------------------------*/
????/*?Reset?of?all?peripherals,?Initializes?the?Flash?interface?and?the?Systick.?*/
????HAL_Init();
????/*?USER?CODE?BEGIN?Init?*/
????/*?USER?CODE?END?Init?*/
????/*?Configure?the?system?clock?*/
????SystemClock_Config();
????/*?USER?CODE?BEGIN?SysInit?*/
????/*?USER?CODE?END?SysInit?*/
????/*?Initialize?all?configured?peripherals?*/
????MX_GPIO_Init();
????MX_I2C1_Init();
????MX_USART1_UART_Init();
????MX_SPI2_Init();
????/*?USER?CODE?BEGIN?2?*/
????/*串口初始化后加這個延時,防止后面的printf打印亂碼*/
????HAL_Delay(200);
????/*注冊驅(qū)動模型*/
????Register_Driver_Model(&lcd_model);
????/*調(diào)用LCD初始化*/
????lcd_model.lcd_driver->lcd_init();
????/*調(diào)用LCD顯示ASCII碼字符串*/
????lcd_model.lcd_driver->lcd_show_ascii_str(ascii_para[0]);
????/*?USER?CODE?END?2?*/
????/*?Infinite?loop?*/
????/*?USER?CODE?BEGIN?WHILE?*/
????while?(1)
????{
????????/*?USER?CODE?END?WHILE?*/
????????/*?USER?CODE?BEGIN?3?*/
????????/*循環(huán)調(diào)用LCD顯示ASCII碼字符串*/
????????for(int?i?=?0?;?i?3?;?i++)
????????{
????????????lcd_model.lcd_driver->lcd_fill(fill_para[i]);
????????????lcd_model.lcd_driver->lcd_show_ascii_str(ascii_para[i]);
????????????HAL_Delay(100);
????????}
????}
????/*?USER?CODE?END?3?*/
}
這樣我們就完成了LCD驅(qū)動最簡單的分層設(shè)計了,當然我們的軟件框架后續(xù)還需要不斷的把它做得更健壯,這樣以后隨便一個LCD,我們都可以設(shè)計一套類似這樣的固定模板,根據(jù)實際的業(yè)務需求,定義設(shè)計合適的接口,以后但凡換一個項目還是用同一個LCD的話就非常簡單了!
5、思考
前面我開源了一個基于TencentOS tiny
的氣體探測儀項目,你是否能在那個項目上繼續(xù)進行優(yōu)化改善呢?
本節(jié)代碼已同步到碼云的代碼倉庫中,獲取方法如下:
1、新建一個文件夾
2、使用git clone遠程獲取小熊派例程存放的代碼倉庫
項目開源倉庫:
https://gitee.com/morixinguan/bear-pi.git
我還將之前做的一些項目以及練習例程在近期內(nèi)全部上傳完畢,與大家一起分享交流:
公眾號粉絲福利時刻
這里我給大家申請到了福利,本公眾號讀者購買小熊派開發(fā)板可享受9折優(yōu)惠,有需要購買小熊派以及騰訊物聯(lián)網(wǎng)開發(fā)板的朋友,淘寶搜索即可,跟客服說你是公眾號:嵌入式云IOT技術(shù)圈?的粉絲,立享9折優(yōu)惠!
往期精彩
軟技能:人人皆可成為卓越的領(lǐng)導!
嵌入式軟件解決ADC電量顯示問題經(jīng)驗分享
一口氣搞懂「文件系統(tǒng)」,就靠這 25 張圖了
做了這么久的技術(shù)自媒體,楊工,您有什么感想要分享的嗎?
覺得本次分享的文章對您有幫助,隨手點[在看]
并轉(zhuǎn)發(fā)分享,也是對我的支持。
免責聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!