日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當前位置:首頁 > 嵌入式 > 嵌入式大雜燴
[導讀]前言 前不久,我有位做測試的朋友轉(zhuǎn)去做開發(fā)的工作,面試遇到了一個問題,他沒明白,打電話問了我。題目大概就是: 在單片機裸機開發(fā)時,單片機要處理多個任務,此時你的程序框架是怎樣的呢? 這其實是個經(jīng)典面試問題,我以前面試也被問過。 答案一:輪詢系統(tǒng)

前言

前不久,我有位做測試的朋友轉(zhuǎn)去做開發(fā)的工作,面試遇到了一個問題,他沒明白,打電話問了我。題目大概就是:

單片機裸機開發(fā)時,單片機要處理多個任務,此時你的程序框架是怎樣的呢?

這其實是個經(jīng)典面試問題,我以前面試也被問過。

答案一:輪詢系統(tǒng)

代碼結(jié)構(gòu)如:

左右滑動查看全部代碼>>>

int?main(void)
{
?init_something();
?
?while(1)
?{
??do_something1();
????????do_something2();
????????do_something3();
?}
}

這種結(jié)構(gòu)大概是我們初學單片機的時候的代碼結(jié)構(gòu)。在沒有外部事件驅(qū)動時,可以較好使用。

只答出了這種情況,印象分估計會比較低,多半涼涼。

答案二:前后臺系統(tǒng)

代碼結(jié)構(gòu)如(該代碼來自 《RT-Thread內(nèi)核實現(xiàn)與應用開發(fā)實踐指南》 ):

左右滑動查看全部代碼>>>

int?flag1?=?0;
int?flag2?=?0;
int?flag3?=?0;

int?main(void)
{
?/*?硬件相關(guān)初始化?*/
?HardWareInit();

?/*?無限循環(huán)?*/
?for?(;;)?{
???if?(flag1)?{
?????/*?處理事情?1?*/
?????DoSomething1();
???}

???if?(flag2)?{
?????/*?處理事情?2?*/
?????DoSomethingg2();
???}

???if?(flag3)?{
?????/*?處理事情?3?*/
?????DoSomethingg3();
???}
?}
}

void?ISR1(void)
{
?/*?置位標志位?*/
?flag1?=?1;
?/*?如果事件處理時間很短,則在中斷里面處理
?如果事件處理時間比較長,在回到后臺處理?*/

?DoSomething1();
}

void?ISR2(void)
{
?/*?置位標志位?*/
?flag2?=?2;

?/*?如果事件處理時間很短,則在中斷里面處理
?如果事件處理時間比較長,在回到后臺處理?*/

?DoSomething2();
}

void?ISR3(void)
{
?/*?置位標志位?*/
?flag3?=?1;
?/*?如果事件處理時間很短,則在中斷里面處理
?如果事件處理時間比較長,在回到后臺處理?*/

?DoSomething3();
}

此處,中斷稱為前臺,main中的while循環(huán)稱為后臺。相比于循環(huán)系統(tǒng),這種方式相對可以提高外部事件的實時響應能力。

可以回答出這種情況,印象分大概一半以上,會再細問。

答案三:升級版前后臺系統(tǒng)(軟件定時器法)

以前,學C語言時,常常聽到有人說:指針是C語言的靈魂,沒學會指針就是沒學會C語言。。

后來,學單片機時,又聽到有人說:中斷和定時器是單片機的靈魂,沒掌握中斷與定時器就沒學會單片機。。

大佬們都那么說了,那就拿定時器來搞點事情。定時器渾身都是寶,本篇筆記我們來介紹使用定時器(系統(tǒng)滴答定時器或者其它定時器)來做的裸機框架。軟件定時器法也有另一種說法:時間片輪詢法。

可以回答出這種情況,這場面試多半穩(wěn)了。

下面以STM32單片機為例看看這種方法的使用。

站在巨人的肩膀上

開源項目—— MultiTimer ,項目倉庫地址:

https://github.com/0x1abin/MultiTimer

1、MultiTimer 簡介

MultiTimer 是一個軟件定時器擴展模塊,可無限擴展你所需的定時器任務,取代傳統(tǒng)的標志位判斷方式, 更優(yōu)雅更便捷地管理程序的時間觸發(fā)時序。

2、MultiTimer 的demo

左右滑動查看全部代碼>>>

#include?"multi_timer.h"

struct?Timer?timer1;
struct?Timer?timer2;

void?timer1_callback()
{
????printf("timer1?timeout!\r\n");
}

void?timer2_callback()
{
????printf("timer2?timeout!\r\n");
}

int?main()
{
????timer_init(&timer1,?timer1_callback,?1000,?1000);?//1s?loop
????timer_start(&timer1);
????
????timer_init(&timer2,?timer2_callback,?50,?0);?//50ms?delay
????timer_start(&timer2);
????
????while(1)?{
????????
????????timer_loop();
????}
}

void?HAL_SYSTICK_Callback(void)
{
????timer_ticks();?//1ms?ticks
}

3、MultiTimer 的移植、剖析

想要對MultiTimer 進行深入學習可閱讀項目源碼及如下這篇文章:

第6期 | MultiTimer,一款可無限擴展的軟件定時器

自己動手,豐衣足食

1、代碼模板

準備一個定時器,可以是系統(tǒng)滴答定時器,也可以是TIM定時器,使用這個定時器拓展出多個軟件定時器。

比如我們系統(tǒng)中有三個任務:LED翻轉(zhuǎn)、溫度采集、溫度顯示。此時我們可以使用一個硬件定時器拓展出3個軟件定時器,定義如下宏定義:

左右滑動查看全部代碼>>>

#define??MAX_TIMER????????????3????????????//?最大定時器個數(shù)
EXT?volatile?unsigned?long????g_Timer1[MAX_TIMER];?
#define??LedTimer?????????????g_Timer1[0]??//?LED翻轉(zhuǎn)定時器
#define??GetTemperatureTimer??g_Timer1[1]??//?溫度采集定時器
#define??SendToLcdTimer???????g_Timer1[2]??//?溫度顯示定時器

#define??TIMER1_SEC????????(1)??????????????//?秒
#define??TIMER1_MIN????????(TIMER1_SEC*60)??//?分


在定時器初始化的時候也順便給三個軟件定時器進行初始化操作:

左右滑動查看全部代碼>>>

/********************************************************************************************************
**?函數(shù):?TIM1_Init,?通用定時器1初始化
**------------------------------------------------------------------------------------------------------
**?參數(shù): arr:自動重裝值 psc:時鐘預分頻數(shù)
**?說明:?定時器溢出時間計算方法:Tout=((arr+1)*(psc+1))/Ft
**?返回:?void?
********************************************************************************************************/

void?TIM1_Init(uint16_t?arr,?uint16_t?psc)
{
????TIM_TimeBaseInitTypeDef??TIM_TimeBaseStructure;
?NVIC_InitTypeDef?NVIC_InitStructure;
?
?RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1,?ENABLE);?
?
?/*?定時器TIM1初始化?*/
?TIM_TimeBaseStructure.TIM_Period?=?arr;?
?TIM_TimeBaseStructure.TIM_Prescaler?=psc;?
?TIM_TimeBaseStructure.TIM_ClockDivision?=?TIM_CKD_DIV1;?
?TIM_TimeBaseStructure.TIM_CounterMode?=?TIM_CounterMode_Up;??
?TIM_TimeBaseStructure.TIM_RepetitionCounter=0;
?TIM_TimeBaseInit(TIM1,?&TIM_TimeBaseStructure);?
??TIM_ClearFlag(TIM1,TIM_FLAG_Update?);
?
?/*?中斷使能?*/
?TIM_ITConfig(TIM1,TIM_IT_Update,?ENABLE?);?
?
?/*?中斷優(yōu)先級NVIC設(shè)置?*/
????NVIC_InitStructure.NVIC_IRQChannel?=??TIM1_UP_IRQn;
?NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority?=?1;??
?NVIC_InitStructure.NVIC_IRQChannelSubPriority?=?0;??
?NVIC_InitStructure.NVIC_IRQChannelCmd?=?ENABLE;
?NVIC_Init(&NVIC_InitStructure);??
?TIM_Cmd(TIM1,?ENABLE);??
????
?//?全局定時器初始化
?for(int?i?=?0;?i??{
??g_Timer1[i]?=?0;???
?}
}


在定時器中斷中對這些軟件定時器進行定時值做遞減操作:

左右滑動查看全部代碼>>>

/********************************************************************************************************
**?函數(shù):?TIM1_IRQHandler,??定時器1中斷服務程序
**------------------------------------------------------------------------------------------------------
**?參數(shù):?無
**?返回:?無?
********************************************************************************************************/

void?TIM1_UP_IRQHandler(void)???//TIM1中斷
{
?uint8?i;
?
?if?(TIM_GetITStatus(TIM1,?TIM_IT_Update)?!=?RESET)??//?檢查TIM1更新中斷發(fā)生與否
?{
??//-------------------------------------------------------------------------------
??//?各種定時間器計時
??for?(i?=?0;?i?//?定時時間遞減?????
???if(?g_Timer1[i]?)?g_Timer1[i]--?;
??TIM_ClearITPendingBit(TIM1,?TIM_IT_Update);?????//清除TIMx更新中斷標志?
?}
}?


我們在各個定時任務中給這些軟件定時器賦予定時值,這些定時值遞減到0則該任務會被觸發(fā)執(zhí)行,比如:

左右滑動查看全部代碼>>>

void?Task_Led(void)
{
?//----------------------------------------------------------------
?//?等待定時時間
?if(LedTimer)?return;
?LedTimer?=?1?*?TIMER1_SEC;
?//----------------------------------------------------------------
?//?LED任務主體
?LedToggle();
}

void?Task_GetTemperature(void)
{
?//----------------------------------------------------------------
?//?等待定時時間
?if(LedTimer)?return;
?LedTimer?=?2?*?TIMER1_SEC;
?//----------------------------------------------------------------
?//?溫度采集任務主體
?GetTemperature();
}

void?Task_SendToLcd(void)
{
?//----------------------------------------------------------------
?//?等待定時時間
?if(LedTimer)?return;
?LedTimer?=?2?*?TIMER1_SEC;
?//----------------------------------------------------------------
?//?溫度顯示任務主體
?LcdDisplay();
}


如此一來,每過1、2、4秒則分別觸發(fā)LED翻轉(zhuǎn)任務、溫度采集任務、溫度顯示任務。

這里配置的最小定時單位為1秒,當然根據(jù)實際需要進行配置(定時器初始化),定時器初始化可以放在系統(tǒng)統(tǒng)一初始化函數(shù)里:

左右滑動查看全部代碼>>>

/********************************************************************************************************
**?函數(shù):?SysInit,?系統(tǒng)上電初始化
**------------------------------------------------------------------------------------------------------
**?參數(shù):?
**?說明:?
**?返回:?
********************************************************************************************************/

void?SysInit(void)
{
?CpuInit();??????????????????//?配置系統(tǒng)信息函數(shù)
?SysTickInit();??????????????//?系統(tǒng)滴答定時器初始化函數(shù)
?UsartInit(115200);??????????//?串口初始化函數(shù),波特率115200
?TIM1_Init(2000-1,?36000-1);?//?定時周期1s
?LedInit();??????????????????//?Led初始化
?TemperatureInit();??????????//?溫度傳感器初始化
?LcdInit();??????????????????//?LCD初始化
}


此時我們的main函數(shù)就可以設(shè)計為:

int?main(void)
{
?//-----------------------------------------------------------------------------------------------?
?//?上電初始化函數(shù)
?SysInit();?
?
?//-----------------------------------------------------------------------------------------------?
?//?主程序
?while?(1)
?{
??//-----------------------------------------------------------------------------------------------?
??//?定時任務
??Task_Led();
??Task_GetTemperature();?
??Task_SendToLcd();
?}
}

主函數(shù)主要是進行系統(tǒng)上電的一些初始化操作,接著是調(diào)用各定時任務函數(shù)。

本demo使用定時器1來擴展出3個軟件定時器,如果TIM資源不夠用,可以換用系統(tǒng)滴答定時器來做。如:

其中,時間基數(shù)可以根據(jù)實際需要進行調(diào)整。

2、實踐(代入法)

套用以上模板,分享我的一個實例:


需要思考及注意的問題是給每個任務的定時值設(shè)置多大合適?這也是一些朋友有疑問的,這只能是自己對自己的任務做考慮,具體情況具體分析,給經(jīng)驗值、調(diào)試調(diào)整。

就如同常常有人問定義多大的數(shù)組合適?在使用RTOS時每個線程的線程棧大小設(shè)置多大合適、優(yōu)先級設(shè)置為多少合適?這些都是需要我們自己進行思考的。

有模板/輪子套用是好事,但有些問題不能單單依靠模板,否則有可能把自己給套進去。

以上是以STM32為例的,其它單片機也是可以用這樣子的思想的,包括51單片機。

面對文首提到的面試問題,若是可以提到使用軟件定時器來處理,進一步能清楚地表達出來,再進一步能寫出一些偽代碼,那這場面試多半是穩(wěn)了。

不僅僅是為了面試,本文的方法是很經(jīng)典的,小編曾經(jīng)接觸的產(chǎn)品項目中就有用到,很實用,值得學習掌握。方法掌握多了,實際應用的時候想用屠龍刀還是倚天劍根據(jù)實際情況選擇使用即可。

以上就是本次的分享,如有錯誤,歡迎指出,謝謝。


猜你喜歡

bug解決不了?使用日志法

單片機工程師的角度看嵌入式Linux


免責聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除( 郵箱:macysun@21ic.com )。
換一批
延伸閱讀

在單片機通信系統(tǒng)中,數(shù)據(jù)幀是實現(xiàn)設(shè)備間可靠對話的核心載體。不同于網(wǎng)絡通信中成熟的TCP/IP協(xié)議,單片機通信往往需要自定義數(shù)據(jù)幀格式,而幀頭、幀尾與校驗機制則是保障數(shù)據(jù)傳輸準確性的三大關(guān)鍵。

關(guān)鍵字: 單片機 數(shù)據(jù)幀

隨著嵌入式技術(shù)的不斷發(fā)展,時序分析工具和方法也在不斷進步,未來將朝著智能化、自動化的方向發(fā)展,為開發(fā)者提供更高效的調(diào)試手段。但無論技術(shù)如何發(fā)展,扎實的時序分析基礎(chǔ)都是嵌入式開發(fā)者不可或缺的能力,只有深入理解通信時序的本質(zhì)...

關(guān)鍵字: 時序 單片機

在嵌入式系統(tǒng)設(shè)計中,不同架構(gòu)、不同廠商的單片機協(xié)同工作早已成為常態(tài)。從8位的51系列到32位的STM32,從精簡指令集的PIC到復雜指令集的AVR,這些性能各異的單片機如何突破硬件差異實現(xiàn)數(shù)據(jù)交互,是嵌入式開發(fā)中的核心課...

關(guān)鍵字: 嵌入式 單片機

在嵌入式系統(tǒng)開發(fā)中,單片機的時鐘系統(tǒng)是整個系統(tǒng)的"心臟",所有的指令執(zhí)行、外設(shè)操作、定時器中斷都依賴于精準的時鐘信號。但在實際開發(fā)過程中,很多開發(fā)者都會遇到單片機時鐘不準的問題,表現(xiàn)為定時器計時偏差、UART通信波特率錯...

關(guān)鍵字: 控制系統(tǒng) 單片機

在單片機開發(fā)與調(diào)試過程中,復位電路作為保障芯片正常啟動的核心模塊,其穩(wěn)定性直接影響程序燒錄與系統(tǒng)運行。實際應用中,不少開發(fā)者會遇到“接穩(wěn)壓電源可正常燒錄,接入電池后卻無法燒錄程序”的故障,此類問題多與復位電路設(shè)計、電池供...

關(guān)鍵字: 單片機 復位電路 時序匹配

在單片機的世界里,“字節(jié)”(Byte)是一個貫穿始終的核心概念。從存儲數(shù)據(jù)到執(zhí)行指令,從變量定義到外設(shè)通信,字節(jié)無處不在。很多初學者在學習單片機時,往往更關(guān)注復雜的程序邏輯和外設(shè)驅(qū)動,卻忽略了字節(jié)這個基礎(chǔ)知識點,導致在后...

關(guān)鍵字: 單片機 字節(jié)

在單片機的數(shù)字邏輯中,我們通常最關(guān)注的是高電平和低電平兩種狀態(tài),它們構(gòu)成了二進制數(shù)字世界的基礎(chǔ)。然而,除了這兩種狀態(tài)之外,還有一種至關(guān)重要但常常被忽視的狀態(tài)——高阻態(tài)(High Impedance State)。高阻態(tài)就...

關(guān)鍵字: 單片機 高阻態(tài)

對于PIC入門者來說,不需要盲目追求高端開發(fā)板,一塊功能均衡、資料豐富的入門款就能滿足需求。比如Microchip官方推出的PIC16F84A開發(fā)板,它搭載經(jīng)典的8位PIC內(nèi)核,引腳布局清晰,自帶LED、按鍵等基礎(chǔ)外設(shè),...

關(guān)鍵字: PIC 單片機

該低功耗器件支持5V運行,在實現(xiàn)高性能的同時,能有效保障系統(tǒng)簡潔性與成本效益

關(guān)鍵字: MCU 單片機 工業(yè)自動化
關(guān)閉