送藥小車代碼倉庫:https://gitee.com/lcsc/medical_car
送藥小車數(shù)據(jù)的發(fā)布與訂閱
既然使用了RTOS,就不能像以前裸機一樣全局變量滿天飛了(當然裸機也不能全局變量滿天飛)。
在實時操作系統(tǒng)(RTOS)中,不推薦或不建議使用全局變量來傳遞信息的原因主要有以下幾點:
競爭條件:在多任務環(huán)境下,全局變量會面臨競爭條件的問題。當多個任務同時讀寫同一個全局變量時,會導致數(shù)據(jù)的不一致性和錯誤的結(jié)果。這是因為任務的執(zhí)行是并發(fā)的,無法控制它們的執(zhí)行順序。
數(shù)據(jù)共享和保護:全局變量被所有任務共享,這意味著多個任務可以同時訪問和修改該變量。如果沒有正確的數(shù)據(jù)保護機制,可能會導致數(shù)據(jù)損壞或沖突。
可維護性和調(diào)試困難:使用全局變量傳遞信息可能導致代碼的可維護性和調(diào)試的困難。由于全局變量可以被任何任務修改,追蹤問題的根源和調(diào)試錯誤可能會變得更加困難。
為了避免上述問題,RTOS提供了一些機制來進行任務間的通信和數(shù)據(jù)傳遞,例如:
消息隊列:任務可以通過消息隊列來發(fā)送和接收消息。每個任務有自己的私有消息隊列,通過發(fā)送和接收消息來進行通信,避免了全局變量的競爭條件和數(shù)據(jù)共享問題。
信號量:信號量可以用于同步任務的執(zhí)行和共享資源的訪問。任務可以通過申請和釋放信號量來控制對共享資源的訪問,并確保任務之間的互斥性和同步性。
事件標志組:事件標志組可以用于任務之間的通知和事件觸發(fā)。一個任務可以等待一個或多個事件標志的狀態(tài)變化,并在事件發(fā)生時被喚醒執(zhí)行相應的操作。
通過使用這些RTOS提供的通信機制,可以更安全地在任務之間傳遞信息,避免競爭條件和數(shù)據(jù)共享問題,并提高代碼的可維護性和調(diào)試效率。
在送藥小車這個工程中采用RT-Thread軟件包uMCN,uMCN (Micro Multi-Communication Node) 提供了一種基于發(fā)布/訂閱模式的安全跨線程/進程的通信方式。在系統(tǒng)中,uMCN 被廣泛應用于任務和模塊間的數(shù)據(jù)通信。使用發(fā)布-訂閱(Publish-Subscribe)機制可以提供更靈活和高效的任務間通信方式,支持一對一、一對多、多對一和多對多的通信模式。這種機制可以解決使用全局變量傳遞信息可能帶來的競爭條件和數(shù)據(jù)共享問題。
發(fā)布-訂閱機制基于事件驅(qū)動的思想,其中包含兩個角色:發(fā)布者(Publisher)和訂閱者(Subscriber)。發(fā)布者負責發(fā)布事件或消息,而訂閱者則注冊對感興趣的事件或消息進行訂閱。他提供了一種松散耦合的通信方式,允許任務或模塊之間通過發(fā)布和訂閱消息來進行通信,而不需要直接知道彼此的存在。
在使用發(fā)布-訂閱模式獲取數(shù)據(jù)時,通常涉及以下角色和操作:
發(fā)布者(Publisher):負責生成和發(fā)布數(shù)據(jù)或事件。發(fā)布者將數(shù)據(jù)發(fā)送到一個或多個特定的主題(Topic),而不關心具體的訂閱者。
訂閱者(Subscriber):訂閱者對特定的主題感興趣,并通過訂閱該主題來接收與之相關的數(shù)據(jù)或事件。
主題(Topic):主題是數(shù)據(jù)或事件的分類或標識符,發(fā)布者根據(jù)主題將數(shù)據(jù)發(fā)送到相應的通道,而訂閱者根據(jù)主題來選擇訂閱的數(shù)據(jù)源。
使用發(fā)布-訂閱模式獲取數(shù)據(jù)的好處在于,系統(tǒng)中的任務或模塊之間解耦合,發(fā)布者和訂閱者之間不直接依賴于彼此的存在,從而提高了系統(tǒng)的可擴展性和靈活性。此外,發(fā)布-訂閱模式還能夠支持多對多的通信,允許多個訂閱者同時接收相同的數(shù)據(jù)或事件,實現(xiàn)了信息的分發(fā)和共享。
他的優(yōu)勢如下:
松耦合性:發(fā)布-訂閱機制使任務之間的通信更加松耦合。發(fā)布者不需要直接知道訂閱者的存在,也不需要關心具體的訂閱者數(shù)量和位置。訂閱者只需要訂閱感興趣的事件或消息,從而實現(xiàn)任務之間的解耦。
靈活性:發(fā)布-訂閱機制支持多對多的通信模式,一個發(fā)布者可以有多個訂閱者,一個訂閱者也可以訂閱多個發(fā)布者的事件或消息。這種靈活性使得任務之間可以方便地建立復雜的通信關系。
擴展性:通過發(fā)布-訂閱機制,可以方便地擴展系統(tǒng),添加新的發(fā)布者或訂閱者,而不需要修改現(xiàn)有的任務邏輯。這種擴展性使得系統(tǒng)更具可維護性和可擴展性。
將各個數(shù)據(jù)發(fā)布,并對接上echo輸出函數(shù)后,就可以在RT-Thread的Finsh控制臺獲取到數(shù)據(jù)信息了(下面是通過uMCN獲取電機編碼器1的數(shù)據(jù)):
具體使用參考uMCN的倉庫:
添加新主題
為了添加新的主題 (topic),你需要先創(chuàng)建一個主題內(nèi)容。例如:
typedef struct {
rt_uint32_t a;
float b;
int8_t c[4];
} data_content;
uMCN對主題內(nèi)容的長度和類型沒有限制,所以理論上可以用來傳輸任何類型的消息。
然后你需要使用宏
MCN_DEFINE(name, size)
來定義主題。一般在發(fā)布主題的源文件的頂部定義主題。例如:
MCN_DEFINE(my_topic, sizeof(data_content));
uMCN 支持一個主題擁有多個發(fā)布者和訂閱者。注意同一個主題名字不同被重復定義,不然編譯器會報錯。
mcn_advertise(MCN_ID(my_topic), my_topic_echo);
MCN_ID()
宏根據(jù)主題名獲得樞紐節(jié)點。my_topic_echo
是一個回調(diào)函數(shù),用來打印主題的數(shù)據(jù)。
static int my_topic_echo(void* param)
{
data_content data;
if(mcn_copy_from_hub((McnHub*)param, &data) == FMT_EOK){
printf("a:%d b:%f c:%c %c %c %cn", data.a, data.b, data.c[0], data.c[1], data.c[2], data.c[3]);
return 0;
}
return -1;
}
發(fā)布主題
可以在系統(tǒng)的任意位置使用函數(shù)
mcn_publish()
來發(fā)布一個主題。例如:
data_content my_data = {50, -2.0, {1,2,3,4}};
mcn_publish(MCN_ID(my_topic), &my_data);
審核編輯 黃宇
-
數(shù)據(jù)
+關注
關注
8文章
6837瀏覽量
88754 -
RTOS
+關注
關注
21文章
809瀏覽量
119366 -
開發(fā)板
+關注
關注
25文章
4913瀏覽量
97086
發(fā)布評論請先 登錄
相關推薦
評論