__weak關(guān)鍵字:程序模塊相互獨(dú)立的大殺器
在STM32 HAL庫開發(fā)中,我們經(jīng)常會看到__weak
這個(gè)關(guān)鍵字,到底是什么意思呢?出于這個(gè)好奇心我們來打開KEIL的幫助手冊找到它的出處:
意思就是,它是一個(gè)弱符號,可以用于修飾變量和函數(shù);不過我們經(jīng)??吹降氖菍瘮?shù)的修飾,所以這里我們僅討論下函數(shù)的修飾就可以了,也就是說,當(dāng)一個(gè)函數(shù)前面加上__weak
這樣的修飾符以后,允許用戶在其它文件中定義一個(gè)和__weak
修飾過的一模一樣的函數(shù),最終當(dāng)編譯器編譯的時(shí)候,會選擇用戶定義的函數(shù),如果用戶沒有重新實(shí)現(xiàn)這個(gè)函數(shù),則編譯器就會去執(zhí)行帶__weak
修飾的函數(shù)。
所以在HAL庫里,比如我們經(jīng)常會看到像下面這樣的函數(shù):
__weak?void?HAL_UART_TxHalfCpltCallback(UART_HandleTypeDef?*huart)
{
????UNUSED(huart);
}
__weak?void?HAL_UART_RxCpltCallback(UART_HandleTypeDef?*huart)
{
????UNUSED(huart);
}
....等等....
通常HAL庫源碼里帶__weak
這個(gè)弱函數(shù)很多內(nèi)部都沒有實(shí)現(xiàn),它把主動權(quán)讓給用戶自己根據(jù)自己的需要去定義一個(gè)一模一樣的函數(shù),然后去做自己想做的事情,這里的UNUSED
起到一個(gè)防止編譯器報(bào)警告的作用,原型如下:
#define?UNUSED(X)?(void)X??????/*?To?avoid?gcc/g++?warnings?*/
這樣就非常好了,我們可以用這樣的機(jī)制輕松實(shí)現(xiàn)程序模塊的相互獨(dú)立,如何來實(shí)現(xiàn)呢?我把我最近做的項(xiàng)目做一個(gè)分享,我完成的是一個(gè)金屬檢測傳感器的模塊框架,為了未來能夠不費(fèi)吹灰之力移植到別的STM32的平臺,我是這么來做的:
Metal.h
#define?UNUSED_METAL(X)?(void)X??
/*金屬傳感器數(shù)據(jù)采集結(jié)構(gòu)體*/
typedef?struct
{
????int?Serial_Number?;???????????? /*流水號*/
????uint16_t?Heating_Value?;?????????/*加熱值*/
????uint16_t?Heating_Signal_Value?;??/*信號值*/
????uint16_t?Devalue?;???????????? /*差值*/
}?Metal_Sensor?;
extern?Metal_Sensor?Metal_Sensor_Device?;
extern?Metal_Sensor?Meatl_Sensor_Parse??;
/*注冊金屬傳感器*/
void?Register_Metal_Sensor(void);
/*獲取并解析傳感器數(shù)據(jù)*/
void?Get_Metal_Sensor_Data(char?*data,?Metal_Sensor?*sensor_data);
Metal.c
/*解析金屬數(shù)據(jù)格式結(jié)構(gòu)體*/
typedef?struct
{
????int?Para1?;
????int?Para2?;
????int?Para3?;
????int?Para4?;
}?Parse_Metal_Passage_Value?;
Metal_Sensor?Meatl_Sensor_Parse?;
Metal_Sensor?Metal_Sensor_Device?;
__weak?void?__Register_Metal_Sensor(void);
__weak?void?CallBack_Metal_Logic(Metal_Sensor?*sensor_data);
/*解析傳感器數(shù)據(jù)*/
void?Split_Sensor_Metal_Data(char?*Data,?Parse_Metal_Passage_Value?*Sensor_Data_Info)
{
????char?*temp?=?NULL?;
????Sensor_Data_Info->Para1?=?atoi(Data);
????temp?=?strstr(Data,?"?");
????Sensor_Data_Info->Para2?=?atoi(temp?+?1);
????temp?=?strstr(temp?+?1,?"?");
????Sensor_Data_Info->Para3?=?atoi(temp?+?1);
????temp?=?strstr(temp?+?1,?"?");
????Sensor_Data_Info->Para4?=?atoi(temp?+?1);
????temp?=?strstr(temp?+?1,?"?");
}
/*金屬傳感器注冊*/
void?Register_Metal_Sensor(void)
{
????__Register_Metal_Sensor();
}
void?Get_Metal_Sensor_Data(char?*data,?Metal_Sensor?*sensor_data)
{
????Parse_Metal_Passage_Value?para_0?;
????Split_Sensor_Metal_Data(data,?¶_0);
????sensor_data->Serial_Number????=?para_0.Para1?;
????sensor_data->Heating_Value?=?para_0.Para2?;
????sensor_data->Devalue??=?para_0.Para3?;
????sensor_data->Heating_Signal_Value???=?para_0.Para4?;
????CallBack_Metal_Logic(sensor_data);
}
__weak?void?__Register_Metal_Sensor(void)
{
????UNUSED_METAL(NULL);
}
__weak?void?CallBack_Metal_Logic(Metal_Sensor?*sensor_data)
{
????UNUSED_METAL(sensor_data);
}
這里提供給外面應(yīng)用邏輯兩個(gè)接口,一個(gè)是用戶需要提供注冊金屬傳感器的邏輯,他只需要實(shí)現(xiàn)__Register_Metal_Sensor
函數(shù)就可以完成金屬傳感器的注冊了;另一個(gè)是用戶拿到解析金屬傳感器的數(shù)據(jù)以后去做他自己要做的事情,那么他只需要實(shí)現(xiàn)CallBack_Metal_Logic
這個(gè)函數(shù)就可以了。
然后在另一個(gè)metal_detect_app.c
文件中,直接實(shí)現(xiàn)這兩個(gè)與Metal.c
中一模一樣的函數(shù)即可:
/*注冊金屬傳感器*/
void?__Register_Metal_Sensor(void)
{
????HAL_UART_DMAStop(&huart6);
????memset(Metal_Sensor_Handler.SensorU6Buffer,?0,?SENSOR_U6_BUFFER_SIZE);
????HAL_UART_Receive_DMA(&huart6,?(uint8_t*)Metal_Sensor_Handler.SensorU6Buffer,?SENSOR_U6_BUFFER_SIZE);
????//開啟空閑中斷
????__HAL_UART_ENABLE_IT(&huart6,?UART_IT_IDLE);
}
/*調(diào)用具體的應(yīng)用邏輯*/
void?CallBack_Metal_Logic(Metal_Sensor?*sensor_data)
{
????/*在不同的頁面執(zhí)行不同的邏輯*/
????switch(Flow_Cursor.flow_cursor)
????{
????case?MAIN_PAGE:
????????//實(shí)現(xiàn)應(yīng)用業(yè)務(wù)邏輯
????????break?;
????case?METAL_TEST_PAGE:
????????//實(shí)現(xiàn)應(yīng)用業(yè)務(wù)邏輯
????????break?;
????default:
????????break?;
????}
????/*重新開啟串口DMA接收*/
????memset(Metal_Sensor_Handler.SensorU6Buffer,?0,?SENSOR_U6_BUFFER_SIZE);
????HAL_UART_Receive_DMA(&huart6,?(uint8_t*)Metal_Sensor_Handler.SensorU6Buffer,?SENSOR_U6_BUFFER_SIZE);
}
最后,在我的任務(wù)中做如下調(diào)用:
/*?金屬傳感器線程入口函數(shù)?*/
static?void?metal_thread_entry(void?*parameter)
{
????static?rt_err_t?result;
????/*?創(chuàng)建一個(gè)動態(tài)信號量,初始值是?0?*/
????metal_sem?=?rt_sem_create("mt_sem",?0,?RT_IPC_FLAG_FIFO);
????if?(metal_sem?==?RT_NULL)
????{
????????rt_kprintf("create?mt_sem?failed.\n");
????????return?;
????}
????/*注冊金屬傳感器*/
????Register_Metal_Sensor();
????while?(1)
????{
????????/*獲取金屬傳感器信號量*/
????????result?=?rt_sem_take(metal_sem,?RT_WAITING_FOREVER);
????????if?(RT_EOK?==?result)
????????????Get_Metal_Sensor_Data((char?*)Metal_Sensor_Handler.SensorU6Buffer,?&Meatl_Sensor_Parse);
????}
}
這樣,就輕松的實(shí)現(xiàn)了模塊的相互獨(dú)立了。
往期精彩
如何優(yōu)雅地打印 HEX 數(shù)據(jù)?
讓傳感器數(shù)據(jù)更直觀之LCD曲線顯示
項(xiàng)目資源太緊張了,如何根據(jù)map信息進(jìn)行功能裁剪和優(yōu)化?
整理了很久之前在碼云/Github/CSDN上收藏的嵌入式產(chǎn)品級項(xiàng)目分享開源
覺得本次分享的文章對您有幫助,隨手點(diǎn)[在看]
并轉(zhuǎn)發(fā)分享,也是對我的支持。
免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!