在本站51hei-5板子上做315兆無線解碼和紅外解碼試驗的時候,延時函數的精度很重要,要做到相當精確才可以成功,所以大家一定要掌握.
這也是大家最常在QQ里問我的一個問題,如果從keil里看了c語言的反匯編代碼然后根據晶振和指令計算延時的時間這樣雖然非常的準確但是相當的麻煩而且容易搞錯,我這里介紹一個最簡單的方法.可以驗證你的延時函數
這里用一個例程詳細介紹一下。
過程參考如下:
在編譯器下建立一個新項目,也可以利用已有項目。此過程中需要注意,單片機晶振的選擇,因為for循環里指令的執行時間和晶振有直接關系,本例中晶振使用11.0592M。
編寫一段關于延時的函數,主要利用for循環,代碼如下:
void delay_ms(unsigned int ms)
{
unsigned int i;
unsigned char j;
for(i=0;i<ms;i++)
{
for(j=0;j<200;j++);
for(j=0;j<102;j++);
}
}
其中ms是輸入參數,如果輸入1,就是要求程序延時1ms。
j變量是調整程序運行的時間參數。調整j的數值,使1次循環的時間在1ms。
將此程序編譯通過,然后利用軟件仿真,調整時間。
下面這個sec就是程序運行到現在的這一行所用的時間。
兩次時間差就是延時函數使用的時間,如果與1ms相差比較多,用戶可以調整j參數的值,使延時時間盡量接近1ms。如增大j的值for(j=0;j<105;j++);
此方法得出延時函數,在晶振不同的情況下,延時時間會不準。軟件調試結果,這個程序的延時時間為:1.01779ms,一般的單片機系統中都可以應用。
下面來說說匯編的傳統計算方法:
指令周期、機器周期與時鐘周期
指令周期:CPU執行一條指令所需要的時間稱為指令周期,它是以機器周期為單位的,指令不同,所需的機器周期也不同。
時鐘周期:也稱為振蕩周期,一個時鐘周期 =晶振的倒數。
MCS-51單片機的一個機器周期=6個狀態周期=12個時鐘周期。
MCS-單片機的指令有單字節、雙字節和三字節的,它們的指令周期不盡相同,一個單周期指令包含一個機器周期,即12個時鐘周期,所以一條單周期指令被執行所占時間為12*(1/12000000)=1us。
了解了上面這些我們來看一個例子
;============延時1秒子程序========================
DELAY_1S: ;延時子程序,12M晶振延時1.002035秒
MOV R4,#10
L3: MOV R2 ,#200 ;1指令周期
L1: MOV R3 ,#249 ;1指令周期
L2: DJNZ R3 ,L2 ;2指令周期
DJNZ R2 ,L1 ;2指令周期
DJNZ R4 ,L3 ;2指令周期
RET ;2指令周期
;循環體延時時間: [(249*2+1+2)*200+1+2]*10*12/12000000=1.002030s
;加上ACALL指令和第一條mov指令以及最后一條RET指令算出來整個函數的時間為1.002035s
;===================================================
通常選用的是11.0592MHZ的晶振:
[(249*2+1+2)*200+1+2]*10*12/11059200=1.08727213541666666...S
匯編延時子程序的延時計算問題
對于程序
DELAY: MOV R0,#00H
DELAY1: MOV R1,#0B3H
DJNZ R1,$
DJNZ R0,DELAY1
查指令表可知 MOV一個機器周期,DJNZ 指令需用兩個機器周期,而一個機器周期時間長度為12/11.0592MHz,所以該段程序執行時間為:
((0B3×2+1+2)×256+1)×12÷11059200=100.2789mS
第一層:DJNZ R1,$:執行了B3H次,一次兩個周期,所以為0B3×2;
第二層:MOV R1,#0B3H為一個周期,DJNZ R0,DELAY1為兩個周期,這樣循環一次就是0B3×2+1+2個周期;第二層的執行次數本來是255次,但因為賦首值為0,而DJNZ是先減1,再比較的,所以就應該是256次。
這樣的話,整個循環執行完應該是(0B3×2+1+2)×256+1次。再加上開始賦值這一句,就是((0B3×2+1+2)×256+1)了
還說明一下:
nop指令或者_nop_(); 函數占一個機器周期,
在stc單片機的12T模式下一個機器周期是一個振蕩周期的12分頻,如果你的晶振是12MHZ,那你的一個機器周期就是1微秒.一個nop指令的執行時間也就是1US
當在6T模式(下載的時候可選擇模式)下12M晶振的時候,一個nop就是0.5US了.
支持。。。。。
admin 斑竹 不愧為強人一個,受教了
要求精確的話用匯編
是可以 比較直觀
學習了。。。。。。。。。。。。。。
哎 我學得是 一塌糊涂啊
學著學著都不知道從何學起了
用C編程的方法解釋不夠祥細
延時的初值可以借助軟件工具那樣比較快
好!有用,實用!
延時時間精確計算哦 ,看看。。。。
學習學習,謝謝樓主!
其實用匯編寫也蠻簡單的 1S的延時:
DELAY:MOV R4,,#50
D1:MOV R5,,#50
D2:MOV R6,,#50
DJNZ R6,,$
DJNZ R5,D2
DJNZ R4,D1
RET
樓主這個方法真是太好了,這樣的話即使是c語言延時也能精確到us級別了
版主啊,這個程序中,j的循環次數是(200+102)*ms 還是200*102*ms啊 如果是第一種的話,為什么不直接寫成j=302 呢 然后最里面的括號直接一個循環語句呢?求回答~
謝謝,很不錯的,很經典
學習了,謝謝
學習了,謝謝
其實要想真正的準確,還要加上調用和結束語句的耗時。
要求嚴格時,還要禁止其他中斷,否則一旦被中斷,延時時間準確就無從談起了。
好東西!
#include<reg52.h>
//#include<math.h>
#define uint unsigned int
#define uchar unsigned char
sbit DUAN=P2^6;
sbit WEI=P2^7;
uchar i,tt;
uchar code Temp[]={
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};
/*void delay(uint z)
{
uint x,y;
for(x=z;x>0;x--)
for(y=110;y>0;y--);
}*/
void main()
{
i=0;
tt=0;
TMOD=0x01;//設置定時器0為工作方式1
TH0=(65536-50000)/256;//高8位
TL0=(65536-50000)%256;//低8位
EA=1;//開總中斷
ET0=1;//開定時器0中斷
TR0=1;//起動定時器0
WEI=1;
P0=0;
WEI=0;
DUAN=1;
P0=0x3f;
DUAN=0;
while(1)
{
tt=0;
i++;
if(tt==20)
{
if(i=16)
i=0;
DUAN=1;
P0=Temp;
DUAN=0;
//delay(1000);
}
}
}
void exter0() interrupt 1
{
TH0=(65536-50000)/256;
TL0=(65536-50000)%256;
tt++;
}
大師們幫忙看一下,我這個讓數碼管隔一秒鐘顯示一個數的程序那點出了錯。
輸出HEX文件時提示Program Size: data=11.0 xdata=0 code=106
creating hex file from "wu13-1"...
"wu13-1" - 0 Error(s), 1 Warning(s).
YS: NOP ;延時 1 ~ 65535 毫秒的延時子程序(晶振為6MHz)
YS1: MOV A, #163 ; 調用前須將需要延時的毫秒數放入 DPTR 中
YS2: DEC A ;絕對精準
JNZ YS2 ;
MOV A, DPL ;
CLR C ;
SUBB A, #1 ;
MOV DPL, A ;
MOV A, DPH ;
SUBB A, #0 ;
MOV DPH, A ;
ORL A, DPL ;
JNZ YS1 ;
RET ;
LZ 寫的很細很到位.
受教了!謝謝!
我覺得用計時器延遲也不錯啊
是不是因為計時器比較占用資源?
如果我想做一個報警器不知道晶振頻率該怎么算?
很好,受教了
好得很,謝謝了。
好東西,一直在考慮這個問題,今天解決了。『呛莮
支持,支持啊
樓主 我想問個問題 為什么把a<=200就不行呢
歡迎光臨 (http://www.raoushi.com/bbs/) | Powered by Discuz! X3.1 |