第27節(jié):在定時(shí)中斷里動(dòng)態(tài)掃描數(shù)碼管的程序
從業(yè)將近十年!手把手教你單片機(jī)程序框架 第27講:
開場(chǎng)白:
上一節(jié)講了在主函數(shù)循環(huán)中動(dòng)態(tài)掃描數(shù)碼管的程序,但是該程序有一個(gè)隱患,在一些項(xiàng)目中 ,主函數(shù)循環(huán)中的任務(wù)越多,就意味著在某一瞬間,每顯示一位數(shù)碼管停留的時(shí)間就會(huì)越久,一旦超過(guò)某個(gè)值,會(huì)嚴(yán)重影響顯示的效果。這一節(jié)要教會(huì)大家兩個(gè)知識(shí)點(diǎn):
第一個(gè):如何把動(dòng)態(tài)掃描數(shù)碼管的程序放在定時(shí)中斷里,徹底解決上節(jié)的顯示隱患。
第二個(gè):在定時(shí)中斷里的重裝初始值不能太大,否則動(dòng)態(tài)掃描數(shù)碼管的速度就不夠。我把原來(lái)常用的初始值2000改成了500。
具體內(nèi)容,請(qǐng)看源代碼講解。
(1)硬件平臺(tái):基于朱兆祺51單片機(jī)學(xué)習(xí)板。用兩片74HC595動(dòng)態(tài)驅(qū)動(dòng)八位共陰數(shù)碼管。
(2)實(shí)現(xiàn)功能:
開機(jī)后顯示 8765.4321 的內(nèi)容,注意,其中有一個(gè)小數(shù)點(diǎn)。
(3)源代碼講解如下:
#include "REG52.H"
void initial_myself();
void initial_peripheral();
void delay_short(unsigned int uiDelayShort);
void delay_long(unsigned int uiDelaylong);
//驅(qū)動(dòng)數(shù)碼管的74HC595
void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01);
void display_drive(); //顯示數(shù)碼管字模的驅(qū)動(dòng)函數(shù)
//驅(qū)動(dòng)LED的74HC595
void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01);
void T0_time(); //定時(shí)中斷函數(shù)
sbit beep_dr=P2^7; //蜂鳴器的驅(qū)動(dòng)IO口
sbit led_dr=P3^5; //作為中途暫停指示燈 亮的時(shí)候表示中途暫停
sbit dig_hc595_sh_dr=P2^0; //數(shù)碼管的74HC595程序
sbit dig_hc595_st_dr=P2^1;
sbit dig_hc595_ds_dr=P2^2;
sbit hc595_sh_dr=P2^3; //LED燈的74HC595程序
sbit hc595_st_dr=P2^4;
sbit hc595_ds_dr=P2^5;
unsigned char ucDigShow8; //第8位數(shù)碼管要顯示的內(nèi)容
unsigned char ucDigShow7; //第7位數(shù)碼管要顯示的內(nèi)容
unsigned char ucDigShow6; //第6位數(shù)碼管要顯示的內(nèi)容
unsigned char ucDigShow5; //第5位數(shù)碼管要顯示的內(nèi)容
unsigned char ucDigShow4; //第4位數(shù)碼管要顯示的內(nèi)容
unsigned char ucDigShow3; //第3位數(shù)碼管要顯示的內(nèi)容
unsigned char ucDigShow2; //第2位數(shù)碼管要顯示的內(nèi)容
unsigned char ucDigShow1; //第1位數(shù)碼管要顯示的內(nèi)容
unsigned char ucDigDot8; //數(shù)碼管8的小數(shù)點(diǎn)是否顯示的標(biāo)志
unsigned char ucDigDot7; //數(shù)碼管7的小數(shù)點(diǎn)是否顯示的標(biāo)志
unsigned char ucDigDot6; //數(shù)碼管6的小數(shù)點(diǎn)是否顯示的標(biāo)志
unsigned char ucDigDot5; //數(shù)碼管5的小數(shù)點(diǎn)是否顯示的標(biāo)志
unsigned char ucDigDot4; //數(shù)碼管4的小數(shù)點(diǎn)是否顯示的標(biāo)志
unsigned char ucDigDot3; //數(shù)碼管3的小數(shù)點(diǎn)是否顯示的標(biāo)志
unsigned char ucDigDot2; //數(shù)碼管2的小數(shù)點(diǎn)是否顯示的標(biāo)志
unsigned char ucDigDot1; //數(shù)碼管1的小數(shù)點(diǎn)是否顯示的標(biāo)志
unsigned char ucDigShowTemp=0; //臨時(shí)中間變量
unsigned char ucDisplayDriveStep=1; //動(dòng)態(tài)掃描數(shù)碼管的步驟變量
unsigned char ucDisplayUpdate=1; //更新顯示標(biāo)志
//根據(jù)原理圖得出的共陰數(shù)碼管字模表
code unsigned char dig_table[]=
{
0x3f, //0 序號(hào)0
0x06, //1 序號(hào)1
0x5b, //2 序號(hào)2
0x4f, //3 序號(hào)3
0x66, //4 序號(hào)4
0x6d, //5 序號(hào)5
0x7d, //6 序號(hào)6
0x07, //7 序號(hào)7
0x7f, //8 序號(hào)8
0x6f, //9 序號(hào)9
0x00, //不顯示 序號(hào)10
};
void main()
{
initial_myself();
delay_long(100);
initial_peripheral();
while(1)
{
;
}
}
/* 注釋一:
* 動(dòng)態(tài)驅(qū)動(dòng)數(shù)碼管的原理是,在八位數(shù)碼管中,在任何一個(gè)瞬間,每次只顯示其中一位數(shù)碼管,另外的七個(gè)數(shù)碼管
* 通過(guò)設(shè)置其公共位com為高電平來(lái)關(guān)閉顯示,只要切換畫面的速度足夠快,人的視覺(jué)就分辨不出來(lái),感覺(jué)八個(gè)數(shù)碼管
* 是同時(shí)亮的。以下dig_hc595_drive(xx,yy)函數(shù),其中第一個(gè)形參xx是驅(qū)動(dòng)數(shù)碼管段seg的引腳,第二個(gè)形參yy是驅(qū)動(dòng)
* 數(shù)碼管公共位com的引腳。
*/
void display_drive()
{
//以下程序,如果加一些數(shù)組和移位的元素,還可以壓縮容量。但是鴻哥追求的不是容量,而是清晰的講解思路
switch(ucDisplayDriveStep)
{
case 1: //顯示第1位
ucDigShowTemp=dig_table[ucDigShow1];
if(ucDigDot1==1)
{
ucDigShowTemp=ucDigShowTemp|0x80; //顯示小數(shù)點(diǎn)
}
dig_hc595_drive(ucDigShowTemp,0xfe);
break;
case 2: //顯示第2位
ucDigShowTemp=dig_table[ucDigShow2];
if(ucDigDot2==1)
{
ucDigShowTemp=ucDigShowTemp|0x80; //顯示小數(shù)點(diǎn)
}
dig_hc595_drive(ucDigShowTemp,0xfd);
break;
case 3: //顯示第3位
ucDigShowTemp=dig_table[ucDigShow3];
if(ucDigDot3==1)
{
ucDigShowTemp=ucDigShowTemp|0x80; //顯示小數(shù)點(diǎn)
}
dig_hc595_drive(ucDigShowTemp,0xfb);
break;
case 4: //顯示第4位
ucDigShowTemp=dig_table[ucDigShow4];
if(ucDigDot4==1)
{
ucDigShowTemp=ucDigShowTemp|0x80; //顯示小數(shù)點(diǎn)
}
dig_hc595_drive(ucDigShowTemp,0xf7);
break;
case 5: //顯示第5位
ucDigShowTemp=dig_table[ucDigShow5];
if(ucDigDot5==1)
{
ucDigShowTemp=ucDigShowTemp|0x80; //顯示小數(shù)點(diǎn)
}
dig_hc595_drive(ucDigShowTemp,0xef);
break;
case 6: //顯示第6位
ucDigShowTemp=dig_table[ucDigShow6];
if(ucDigDot6==1)
{
ucDigShowTemp=ucDigShowTemp|0x80; //顯示小數(shù)點(diǎn)
}
dig_hc595_drive(ucDigShowTemp,0xdf);
break;
case 7: //顯示第7位
ucDigShowTemp=dig_table[ucDigShow7];
if(ucDigDot7==1)
{
ucDigShowTemp=ucDigShowTemp|0x80; //顯示小數(shù)點(diǎn)
}
dig_hc595_drive(ucDigShowTemp,0xbf);
break;
case 8: //顯示第8位
ucDigShowTemp=dig_table[ucDigShow8];
if(ucDigDot8==1)
{
ucDigShowTemp=ucDigShowTemp|0x80; //顯示小數(shù)點(diǎn)
}
dig_hc595_drive(ucDigShowTemp,0x7f);
break;
}
ucDisplayDriveStep++;
if(ucDisplayDriveStep>8) //掃描完8個(gè)數(shù)碼管后,重新從第一個(gè)開始掃描
{
ucDisplayDriveStep=1;
}
/* 注釋二:
* 如果直接是單片機(jī)的IO口引腳驅(qū)動(dòng)的數(shù)碼管,由于驅(qū)動(dòng)的速度太快,此處應(yīng)該適當(dāng)增加一點(diǎn)delay延時(shí)或者
* 用計(jì)數(shù)延時(shí)的方式來(lái)延時(shí),目的是在八位數(shù)碼管中切換到每位數(shù)碼管顯示的時(shí)候,都能停留一會(huì)再切換到其它
* 位的數(shù)碼管界面,這樣可以增加顯示的效果。但是,由于朱兆祺51學(xué)習(xí)板是間接經(jīng)過(guò)74HC595驅(qū)動(dòng)數(shù)碼管的,
* 在單片機(jī)驅(qū)動(dòng)74HC595的時(shí)候,dig_hc595_drive函數(shù)本身內(nèi)部需要執(zhí)行很多指令,已經(jīng)相當(dāng)于delay延時(shí)了,
* 因此這里不再需要加delay延時(shí)函數(shù)或者計(jì)數(shù)延時(shí)。
*/
}
//數(shù)碼管的74HC595驅(qū)動(dòng)函數(shù)
void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01)
{
unsigned char i;
unsigned char ucTempData;
dig_hc595_sh_dr=0;
dig_hc595_st_dr=0;
ucTempData=ucDigStatusTemp16_09; //先送高8位
for(i=0;i<8;i++)
{
if(ucTempData>=0x80)dig_hc595_ds_dr=1;
else dig_hc595_ds_dr=0;
/* 注釋三:
* 注意,此處的延時(shí)delay_short必須盡可能小,否則動(dòng)態(tài)掃描數(shù)碼管的速度就不夠。
*/
dig_hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
delay_short(1);
dig_hc595_sh_dr=1;
delay_short(1);
ucTempData=ucTempData<<1;
}
ucTempData=ucDigStatusTemp08_01; //再先送低8位
for(i=0;i<8;i++)
{
if(ucTempData>=0x80)dig_hc595_ds_dr=1;
else dig_hc595_ds_dr=0;
dig_hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
delay_short(1);
dig_hc595_sh_dr=1;
delay_short(1);
ucTempData=ucTempData<<1;
}
dig_hc595_st_dr=0; //ST引腳把兩個(gè)寄存器的數(shù)據(jù)更新輸出到74HC595的輸出引腳上并且鎖存起來(lái)
delay_short(1);
dig_hc595_st_dr=1;
delay_short(1);
dig_hc595_sh_dr=0; //拉低,抗干擾就增強(qiáng)
dig_hc595_st_dr=0;
dig_hc595_ds_dr=0;
}
//LED燈的74HC595驅(qū)動(dòng)函數(shù)
void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01)
{
unsigned char i;
unsigned char ucTempData;
hc595_sh_dr=0;
hc595_st_dr=0;
ucTempData=ucLedStatusTemp16_09; //先送高8位
for(i=0;i<8;i++)
{
if(ucTempData>=0x80)hc595_ds_dr=1;
else hc595_ds_dr=0;
hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
delay_short(1);
hc595_sh_dr=1;
delay_short(1);
ucTempData=ucTempData<<1;
}
ucTempData=ucLedStatusTemp08_01; //再先送低8位
for(i=0;i<8;i++)
{
if(ucTempData>=0x80)hc595_ds_dr=1;
else hc595_ds_dr=0;
hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
delay_short(1);
hc595_sh_dr=1;
delay_short(1);
ucTempData=ucTempData<<1;
}
hc595_st_dr=0; //ST引腳把兩個(gè)寄存器的數(shù)據(jù)更新輸出到74HC595的輸出引腳上并且鎖存起來(lái)
delay_short(1);
hc595_st_dr=1;
delay_short(1);
hc595_sh_dr=0; //拉低,抗干擾就增強(qiáng)
hc595_st_dr=0;
hc595_ds_dr=0;
}
void T0_time() interrupt 1
{
TF0=0; //清除中斷標(biāo)志
TR0=0; //關(guān)中斷
display_drive(); //數(shù)碼管字模的驅(qū)動(dòng)函數(shù)
/* 注釋四:
* 注意,此處的重裝初始值不能太大,否則動(dòng)態(tài)掃描數(shù)碼管的速度就不夠。我把原來(lái)常用的2000改成了500。
*/
TH0=0xfe; //重裝初始值(65535-500)=65035=0xfe0b
TL0=0x0b;
TR0=1; //開中斷
}
void delay_short(unsigned int uiDelayShort)
{
unsigned int i;
for(i=0;i
{
; //一個(gè)分號(hào)相當(dāng)于執(zhí)行一條空語(yǔ)句
}
}
void delay_long(unsigned int uiDelayLong)
{
unsigned int i;
unsigned int j;
for(i=0;i
{
for(j=0;j<500;j++) //內(nèi)嵌循環(huán)的空指令數(shù)量
{
; //一個(gè)分號(hào)相當(dāng)于執(zhí)行一條空語(yǔ)句
}
}
}
void initial_myself() //第一區(qū) 初始化單片機(jī)
{
led_dr=0; //關(guān)閉獨(dú)立LED燈
beep_dr=1; //用PNP三極管控制蜂鳴器,輸出高電平時(shí)不叫。
hc595_drive(0x00,0x00); //關(guān)閉所有經(jīng)過(guò)另外兩個(gè)74HC595驅(qū)動(dòng)的LED燈
TMOD=0x01; //設(shè)置定時(shí)器0為工作方式1
TH0=0xfe; //重裝初始值(65535-500)=65035=0xfe0b
TL0=0x0b;
}
void initial_peripheral() //第二區(qū) 初始化外圍
{
/* 注釋五:
* 讓數(shù)碼管顯示的內(nèi)容轉(zhuǎn)移到以下幾個(gè)變量接口上,方便以后編寫更上一層的窗口程序。
* 只要更改以下對(duì)應(yīng)變量的內(nèi)容,就可以顯示你想顯示的數(shù)字。初學(xué)者應(yīng)該仔細(xì)看看display_drive等函數(shù),
* 了解來(lái)龍去脈,就可以知道本驅(qū)動(dòng)程序的框架原理了。
*/
ucDigShow8=8; //第8位數(shù)碼管要顯示的內(nèi)容
ucDigShow7=7; //第7位數(shù)碼管要顯示的內(nèi)容
ucDigShow6=6; //第6位數(shù)碼管要顯示的內(nèi)容
ucDigShow5=5; //第5位數(shù)碼管要顯示的內(nèi)容
ucDigShow4=4; //第4位數(shù)碼管要顯示的內(nèi)容
ucDigShow3=3; //第3位數(shù)碼管要顯示的內(nèi)容
ucDigShow2=2; //第2位數(shù)碼管要顯示的內(nèi)容
ucDigShow1=1; //第1位數(shù)碼管要顯示的內(nèi)容
ucDigDot8=0;
ucDigDot7=0;
ucDigDot6=0;
ucDigDot5=1; //顯示第5位的小數(shù)點(diǎn)
ucDigDot4=0;
ucDigDot3=0;
ucDigDot2=0;
ucDigDot1=0;
EA=1; //開總中斷
ET0=1; //允許定時(shí)中斷
TR0=1; //啟動(dòng)定時(shí)中斷
}
總結(jié)陳詞:
有的朋友會(huì)質(zhì)疑,很多教科書上說(shuō),定時(shí)中斷函數(shù)里面的內(nèi)容應(yīng)該越少越好,你把動(dòng)態(tài)驅(qū)動(dòng)數(shù)碼管的函數(shù)放在中斷里面,難道不會(huì)影響其它任務(wù)的執(zhí)行嗎?我的回答是,大部分的小項(xiàng)目都不會(huì)影響,只有少數(shù)實(shí)時(shí)性要求非常高的項(xiàng)目會(huì)影響,而對(duì)于這類項(xiàng)目,我的做法是從一開始設(shè)計(jì)硬件電路板的時(shí)候,就應(yīng)該放棄用動(dòng)態(tài)掃描數(shù)碼管的方案,而是應(yīng)該選數(shù)碼管專用驅(qū)動(dòng)芯片來(lái)實(shí)現(xiàn)靜態(tài)驅(qū)動(dòng)。因?yàn)閯?dòng)態(tài)掃描數(shù)碼管本來(lái)就不適合應(yīng)用在實(shí)時(shí)性非常高的項(xiàng)目。
前面這兩節(jié)都講了數(shù)碼管的驅(qū)動(dòng)程序,要在此基礎(chǔ)上,做一些項(xiàng)目中經(jīng)常遇到的界面應(yīng)用,我們?cè)撛趺磳懗绦?欲知詳情,請(qǐng)聽(tīng)下回分解-----數(shù)碼管通過(guò)切換窗口來(lái)設(shè)置參數(shù)。