很多新手在單片機上走的第一步是點亮第一個LED燈,實際上因為開發板的不同,所編寫的代碼也不同,關鍵是你要去了解你用的開發板的電路布局。對于電路方面的知識我這里也不詳講,我要做的是無論你用哪一種開發板我的文章都能幫助你。
P0 = 0xFE;
這句代碼大家不陌生。
void main(){
unsigned char count = 0;
while(1){
P0 = ~(0x01 << count);
Delay();//單獨實現一個延時函數
count++;
if(count > =8){
count = 0;
}
}
}
以上就是實現流水燈的基本代碼,這里沒有電路供你分析,但是無論什么開發板,核心代碼可以用以上代碼實現。
我相信你能看到這里也是有點基礎的,這里的延時函數Delay,接下來要講的是定時器,定時器就是可以替代延時函數的。
定時器
標準的51單片機內部有T0和T1兩個定時器,實際上就是TCON特殊功能的寄存器來控制這兩個定時器的。
除此之外,定時值存儲寄存器有TH和TL,給TL賦值后,TL會自動加1,加到255后TH加1,有趣的TH也可以提前賦值,但這只是定時器工作的一種模式,定時器有四種模式,這里我不祥講,而且我們幾乎用的模式就是這種,后面涉及到會詳細講解。這里只需要知道TCON(地址0x88)位分配,以后會經常用到。
還有一個TMOC就是定時器作用的模式,位分配如下圖:
代碼:
void main()
{TH0 = 0xB8; //給TH0賦值,后面的0代表是給定時器T0的TH賦值
TL0 = 0x00;
TR0 = 1;//啟動T0定時器
if(TF0 == 1) //判斷T0是否溢出,TF是個標志位
{ //重置
TH0 = 0xB8;
TL0 = 0x00;}}
以上就是定時器,時間多少呢?
我們以晶振位11.0592為例,時鐘周期是1/11059200,機器周期(1ms)12/11059200,如果我們定時20ms,那個要執行20*(12/110592)次,算出來是18432次,換成十六進制是B800,所以對TH0賦值B8,對TL0賦值00;
#include
sbit ADDR0 = P1^0;
sbit ADDR1 = P1^1;
sbit ADDR2 = P1^2;
sbit ADDR3 = P1^3;
sbit ENLED = P1^4;
void main() {
ADDR2 = 1;
ADDR1 = 0;
ADDR0 = 1;
ADDR3 = 1;
ENLED = 0;
P0 = 0XF8;
while(1);}
上面代碼是用位STC-51開發板寫的,在最后一個數碼管上顯示數字7,數碼管難度簡單,只需要針對數碼管等的排布編程即可。
下面我們用關鍵字code定義數碼管所能夠顯示所有字符的數組,這里再結合定時器一起。
#include
//數組
unsigned char code led[] =
{ 0xC0, 0xF9, 0xA4,
0xB0, 0x99, 0x92,
0x82, 0xF8, 0x80,
0x90, 0x88, 0x83,
0xC6, 0xA1, 0x86,0x8E};
void main() { unsigned char count = 0;
//記錄T0中斷次數
unsigned char secnt = 0;
//記錄經過的秒數
ADDR2 = 1;
ADDR1 = 0;
ADDR0 = 0;
ADDR3 = 1;
ENLED = 0;
//設置T0模式
TMOD = 0x01;
//為T0的TH0,TL0初始化
TH0 = 0xB8;
TL0 = 0x00;
//啟動T0 TR0 = 1;
while(1){
if(TF0 ==1)
{ TH0 = 0xB8;
TL0 = 0x00;
count++;
TF0 = 0; }
if(count >=50)
{ count = 0;
P0 = led[secnt];
secnt++;
if(secnt>=16)
{ secnt = 0; } } }}
這里代碼比較緊湊,不過不影響。上面的代碼我相信你也能懂,但是你能發現定時器在這里起到了一個定時中斷的作用。
這里講一下中斷。
中斷
下面是中斷IE寄存器位分配圖:
直接上代碼:
#include
//數碼管顯示字符真值數組
unsigned char code ledchar[]=
{0xC0, 0xF9, 0xA4, 0xB0, 0x99,
0x92, 0x82, 0xF8, 0x80, 0x90,
0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E };
//數碼管顯示區數組
unsigned char ledbuff[6] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
sbit ADDR0 = P1^0;sbit ADDR1 = P1^1;sbit ADDR2 = P1^2;sbit ADDR3 = P1^3;sbit ENLED = P1^4;
unsigned char i = 0;
//動態掃描索引
unsigned int c = 0;
//記錄中斷次數
void main() {
unsigned long s = 0;
//記錄秒數
//使能U3
ADDR3 = 1;
ENLED = 0;
//設置T0模式
TMOD = 0x01;
//初始化TH0,TL0
TH0 = 0xFC; TL0 = 0x66;
//啟動TR0
TR0 = 1;
//使能總中斷
EA = 1;
//使能T0中斷
ET0 = 1;
//主循環
while(1) { //1s中斷 if(c>=1000) { s++; c=0; //為數碼管顯示區賦值 ledbuff[0] = ledchar[s%10]; ledbuff[1] = ledchar[s/10%10]; ledbuff[2] = ledchar[s/100%10]; ledbuff[3] = ledchar[s/1000%10]; ledbuff[4] = ledchar[s/10000%10]; ledbuff[5] = ledchar[s/100000%10]; } }}
//定時器T0中斷服務void InterruptTimer0() interrupt 1{ //重新賦值 TH0 = 0xFC; TL0 = 0x66; c++; //顯示消隱 P0 = 0xFE; //完成數碼管動態掃描 switch(i) { case 0: ADDR2 = 1;ADDR1 = 0; ADDR0 = 1; i++; P0 = ledbuff[0]; break;
case 1: ADDR2 = 1;ADDR1 = 0; ADDR0 = 0;i++; P0 = ledbuff[1]; break;
case 2: ADDR2 = 0;ADDR1 = 1; ADDR0 = 1;i++; P0 = ledbuff[2]; break;
case 3: ADDR2 = 0;ADDR1 = 1; ADDR0 = 0; i++; P0 = ledbuff[3]; break;
case 4: ADDR2 = 0; ADDR1 = 0; ADDR0 = 1;i++; P0 = ledbuff[4]; break;
case 5: ADDR2 = 0;ADDR1 = 0; ADDR0 = 0;i=0; P0 = ledbuff[5]; break; default:break;
}
}
這組代碼能夠按照我們計算好的時間為單位顯示秒數。
我們能夠提出中斷核心代碼
EA = 1//中斷總開關
ET0 = 1//確認使用T0定時器中斷開關
TR0 = 1//肯定要啟動T0定時器
void InterruptTimer0() interrupt 1//定時器T0中斷服務,中斷代碼寫在這里面,至于interrupt 1是因為interrupt會去尋找地址' 1 ',而T0定時器中斷的地址就是1,所以我們可以直接在此函數中寫中斷期間的代碼。至于各種中斷的地址我也不再這里多寫了。值得一談的是IP——中斷優先級寄存器位分配
各級中斷都差不多,中斷發生的也很多,當同時有許多中斷發生時,可以通過置上面的值為1升級成優先級中斷。
-
數碼管
+關注
關注
32文章
1874瀏覽量
90946 -
定時器
+關注
關注
23文章
3237瀏覽量
114471
原文標題:單片機定時器與數碼管靜態顯示
文章出處:【微信號:mcugeek,微信公眾號:MCU開發加油站】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論