當(dāng)前位置:首頁(yè) > 單片機(jī) > 單片機(jī)
[導(dǎo)讀]看到了個(gè)好帖,我在此在它得基礎(chǔ)上再拋拋磚!有個(gè)好帖,從精度考慮,它得研究結(jié)果是: void delay2(unsigned char i) { while(-i); } 為最佳方法。 分析:假設(shè)外掛12M(之后都是在這基礎(chǔ)上討論) 我編譯了下,傳

看到了個(gè)好帖,我在此在它得基礎(chǔ)上再拋拋磚!

有個(gè)好帖,從精度考慮,它得研究結(jié)果是:
void delay2(unsigned char i)
{
while(-i);
}
為最佳方法。
分析:假設(shè)外掛12M(之后都是在這基礎(chǔ)上討論)
我編譯了下,傳了些參數(shù),并看了匯編代碼,觀察記錄了下面的數(shù)據(jù):
delay2(0):延時(shí)518us 518-2*256=6
delay2(1):延時(shí)7us(原帖寫“5us”是錯(cuò)的,^_^)
delay2(10):延時(shí)25us 25-20=5
delay2(20):延時(shí)45us 45-40=5
delay2(100):延時(shí)205us 205-200=5
delay2(200):延時(shí)405us 405-400=5
見(jiàn)上可得可調(diào)度為2us,而最大誤差為6us。
精度是很高了!
但這個(gè)程序的最大延時(shí)是為518us顯然不
能滿足實(shí)際需要,因?yàn)楹芏鄷r(shí)候需要延遲比較長(zhǎng)的時(shí)間。
那么,接下來(lái)討論將t分配為兩個(gè)字節(jié),即uint型的時(shí)候,會(huì)出現(xiàn)什么情況。
void delay8(uint t)
{
while(-t);
}
我編譯了下,傳了些參數(shù),并看了匯編代碼,觀察記錄了下面的數(shù)據(jù):
delay8(0):延時(shí)524551us 524551-8*65536=263
delay8(1):延時(shí)15us
delay8(10):延時(shí)85us 85-80=5
delay8(100):延時(shí)806us 806-800=6
delay8(1000):延時(shí)8009us 8009-8000=9
delay8(10000):延時(shí)80045us 80045-8000=45
delay8(65535):延時(shí)524542us 524542-524280=262
如果把這個(gè)程序的可調(diào)度看為8us,那么最大誤差為263us,但這個(gè)延時(shí)程序還是不能滿足要求的,因?yàn)檠訒r(shí)最大為524.551ms。
那么用ulong t呢?
一定很恐怖,不用看編譯后的匯編代碼了。。。
那么如何得到比較小的可調(diào)度,可調(diào)范圍大,并占用比較少得RAM呢?請(qǐng)看下面的程序:
/*-
程序名稱:50us延時(shí)
注意事項(xiàng):基于1MIPS,AT89系列對(duì)應(yīng)12M晶振,W77.W78系列對(duì)應(yīng)3M晶振
例子提示:調(diào)用delay_50us(20),得到1ms延時(shí)
全局變量:無(wú)
返回:無(wú)
-*/
void delay_50us(uint t)
{
uchar j;
for(;t>0;t-)
for(j=19;j>0;j-)
;
}
我編譯了下,傳了些參數(shù),并看了匯編代碼,觀察記錄了下面的數(shù)據(jù):
delay_50us(1):延時(shí)63us 63-50=13
delay_50us(10):延時(shí)513us 503-500=13
delay_50us(100):延時(shí)5013us 5013-5000=13
delay_50us(1000):延時(shí)50022us 50022-50000=22
赫赫,延時(shí)50ms,誤差僅僅22us,作為C語(yǔ)言已經(jīng)是可以接受了。再說(shuō)要求再精確的話,就算是用匯編也得改用定時(shí)器了。
/*-
程序名稱:50ms延時(shí)
注意事項(xiàng):基于1MIPS,AT89系列對(duì)應(yīng)12M晶振,W77.W78系列對(duì)應(yīng)3M晶振
例子提示:調(diào)用delay_50ms(20),得到1s延時(shí)
全局變量:無(wú)
返回:無(wú)
-*/
void delay_50ms(uint t)
{
uint j;
/****
可以在此加少許延時(shí)補(bǔ)償,以禰補(bǔ)大數(shù)值傳遞時(shí)(如delay_50ms(1000))造成的誤差,
但付出的代價(jià)是造成傳遞小數(shù)值(delay_50ms(1))造成更大的誤差。
因?yàn)閷?shí)際應(yīng)用更多時(shí)候是傳遞小數(shù)值,所以補(bǔ)建議加補(bǔ)償!
****/
for(;t>0;t-)
for(j=6245;j>0;j-)
;
}
我編譯了下,傳了些參數(shù),并看了匯編代碼,觀察記錄了下面的數(shù)據(jù):
delay_50ms(1):延時(shí)50 010 10us
delay_50ms(10):延時(shí)499 983 17us
delay_50ms(100):延時(shí)4 999 713 287us
delay_50ms(1000):延時(shí)4 997 022 2.978ms
赫赫,延時(shí)50s,誤差僅僅2.978ms,可以接受!
上面程序沒(méi)有才用long,也沒(méi)采用3層以上的循環(huán),而是將延時(shí)分拆為兩個(gè)程序以提高精度。應(yīng)該是比較好的做法了。


C51延時(shí)從精度考慮,下面的方法為最佳方法:
voiddelay(unsignedchari)
{
while(--i);
}

功能:延時(shí)
參數(shù):c的范圍為:0 -255
當(dāng) C=0時(shí),延時(shí)為 2 *256+6=518個(gè)機(jī)器周期
當(dāng)C為其它值時(shí),延時(shí)為 2* C+5個(gè)機(jī)器周期
注釋:① 以上是以常數(shù)調(diào)用的計(jì)算值,如果用變量調(diào)用,根據(jù)變量的存
儲(chǔ)類型的不同可能要增加一到三個(gè)周期
② 編譯環(huán)境:KeilC

voiddelay(unsigned char i)
{
while(--i); //不可寫成i--,否則生成的代碼很難看
}

編譯后的子程序內(nèi)是一條DJNZ指令,耗時(shí)2個(gè)周期,由于準(zhǔn)備到R7中,調(diào)用LCALL,還有返回RET等要消耗時(shí)間,所以要加上5個(gè)周期。
用0調(diào)用時(shí)有點(diǎn)特別,0減1后變作0xFF,所以循環(huán)次數(shù)為256次,另外一點(diǎn)很有意思:
當(dāng)用 1-255的常數(shù)調(diào)用時(shí),生成的代碼如下

MOVR7, #1 ; #1 - 255
LCALL _Delay

0 調(diào)用則多了條指令

CLR A
MOV R7, A
LCALL _Delay
所以參數(shù)為 0 時(shí),要多加一個(gè)周期

用變量調(diào)用時(shí)(如Delay(i)之類的形式)根據(jù)變量i的存儲(chǔ)類型加上相應(yīng)的訪問(wèn)時(shí)間, 該函數(shù)的分辨率為 2個(gè)機(jī)器周期,最小延時(shí)為7個(gè)機(jī)器周期。
寫延時(shí)程序時(shí)一定要考慮系統(tǒng)使用的晶振的頻率。對(duì)于你所說(shuō)的微秒級(jí)的延時(shí),如果系統(tǒng)使用12MHZ的晶振,則一條NOP匯編指令便是延時(shí)1微秒。

在做ms級(jí)以下的延時(shí)時(shí),一定要用匯編,那樣才能夠很精確的控制!其他主控程序可以使用c語(yǔ)言!

在C中嵌入?yún)R編用下面格式
_asm
{
//在這里加入?yún)R編的程序段
}

如果是單片機(jī),我想可以用_nop_指令!一個(gè)_nop指令是12*1/12MHz=1us,所以,當(dāng)然如果采用循環(huán)這也會(huì)消耗掉一定的時(shí)間,可以實(shí)機(jī)調(diào)試,用示波器觀察!

有些特殊的應(yīng)用會(huì)用到比較精確的延時(shí)(比如DS18B20等),而C不像匯編,延時(shí)精準(zhǔn)度不好算。本人經(jīng)過(guò)反復(fù)調(diào)試,對(duì)照KEIL編譯后的匯編源文件,得出了以下幾條精確延時(shí)的語(yǔ)句(絕對(duì)精確!本人已通過(guò)實(shí)際測(cè)試),今天貼上來(lái),希望對(duì)需要的朋友有所幫助。

sbitLED=P1^0;//定義一個(gè)管腳(延時(shí)測(cè)試用)
unsignedinti=3;//注意i,j的數(shù)據(jù)類型,
unsignedcharj=3;//不同的數(shù)據(jù)類型延時(shí)有很大不同
//-----------------各種精確延時(shí)語(yǔ)句-----------------------------------
while((i--)!=1);//延時(shí)10*i個(gè)機(jī)器周期
i=10;while(--i);//延時(shí)8*i+2個(gè)機(jī)器周期
i=10;while(i--);//延時(shí)(i+1)*9+2個(gè)機(jī)器周期
j=5;while(--j);//延時(shí)2*j+1個(gè)機(jī)器周期
j=5;while(j--);//延時(shí)(j+1)*6+1個(gè)機(jī)器周期

i=5;
while(--i)//延時(shí)i*10+2個(gè)機(jī)器周期,在i*10+2個(gè)機(jī)器周期
if(LED==0)break;//內(nèi)檢測(cè)到LED管腳為低電平時(shí)跳出延時(shí)

i=5;
while(LED)//每隔10個(gè)機(jī)器周期檢測(cè)一次LED管腳狀態(tài),當(dāng)LED
if((--i)==0)break;//為低時(shí)或者到了10*i+2個(gè)機(jī)器周期時(shí)跳出延時(shí)
//--------------------------------------------------------------------

例如18b20的復(fù)位函數(shù)(12M晶振):
//***********************************************************************
//函數(shù)功能:18B20復(fù)位
//入口參數(shù):無(wú)
//出口參數(shù):unsignedcharx:0:成功1:失敗
//***********************************************************************
unsignedcharow_reset(void)
{
unsignedcharx=0;//12M晶振1個(gè)機(jī)器周期為1us
DQ=1; //DQ復(fù)位
j=10;while(--j);//稍做延時(shí)(延時(shí)10*2+1=21個(gè)機(jī)器周期,21us)
DQ=0; //單片機(jī)將DQ拉低
j=85;while(j--);//精確延時(shí)(大于480us)85*6+1=511us
DQ=1; //拉高總線
j=10;while(j--);//精確延時(shí)10*6+1=61us
x=DQ; //稍做延時(shí)后,
returnx; //如果x=0則初始化成功x=1則初始化失敗
j=25;while(j--);//精確延時(shí)25*6+1=151us
}
//*********************************************************************************
再如紅外解碼程序:
(先說(shuō)傳統(tǒng)紅外解碼的弊端:
程序中用了while(IR_IO);while(!IR_IO);這樣的死循環(huán),如果管腳一直處于一種狀態(tài),就會(huì)一直執(zhí)行while,造成“死機(jī)”現(xiàn)象。當(dāng)然這種情況很少,但我們也的考慮到。而用以下程序則不會(huì),在規(guī)定的時(shí)間內(nèi)沒(méi)有正確的電平信號(hào)就會(huì)返回主程序,這樣就不會(huì)出現(xiàn)“死機(jī)”了)
//***************************外部中斷0*******************************
voidint0(void)interrupt0
{
unsignedchari,j;
unsignedintcount=800;
//--------------8.5ms低電平引導(dǎo)碼-------------------------------------
while(--count)
if(IR_IO==1)return;//在小于8ms內(nèi)出現(xiàn)高電平,返回
count=100;//延時(shí)1ms
while(!IR_IO)//等待高電平
if((--count)==0)return;//在9ms內(nèi)未出現(xiàn)高電平,返回
//-------------4.5ms高電平引導(dǎo)碼------------------------------------
count=410;//延時(shí)4.1ms
while(--count)//...
if(IR_IO==0)return;//在4.1ms內(nèi)出現(xiàn)低電平,返回
count=50;//延時(shí)0.5ms
while(IR_IO)//等待低電平
if((--count)==0)return;//在4.7ms內(nèi)未出現(xiàn)低電平,返回
//-----------------------------------------------------------------
//------------4個(gè)數(shù)據(jù)碼------------------------------------
for(j=0;j<4;j++)
{
for(i=0;i<8;i++)
{
IR_data[j]<<=1;//裝入數(shù)據(jù)
count=60;//延時(shí)0.6ms
while(!IR_IO)//等待高電平
if((--count)==0)return;//在0.6ms內(nèi)未出現(xiàn)高電平,返回
count=40;//低電平結(jié)束,繼續(xù)
while(--count)//延時(shí)0.4ms
if(IR_IO==0)return;//在0.4ms內(nèi)出現(xiàn)低電平,返回
count=100;//延時(shí)1.4ms
while(IR_IO)//檢測(cè)IO狀態(tài)
if((--count)==0)//等待1.4ms到來(lái)
{//在1.4ms內(nèi)都是高電平
IR_data[j]|=1;//兩個(gè)單位高電平,為數(shù)據(jù)1
break;//跳出循環(huán)
}
count=20;//延時(shí)0.2ms
while(IR_IO)//等待低電平跳出
if((--count)==0)return;//0.2ms內(nèi)未出現(xiàn)低電平,返回
}
}
//-------------------------------------------------------------------
flag_IR=1;//置位紅外接收成功標(biāo)志
}

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

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開(kāi)發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開(kāi)幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉