新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 第27節(jié):在定時(shí)中斷里動(dòng)態(tài)掃描數(shù)碼管的程序

第27節(jié):在定時(shí)中斷里動(dòng)態(tài)掃描數(shù)碼管的程序

作者: 時(shí)間:2016-11-22 來源:網(wǎng)絡(luò) 收藏
開場(chǎng)白:

上一節(jié)講了在主函數(shù)循環(huán)中動(dòng)態(tài)掃描數(shù)碼管的程序,但是該程序有一個(gè)隱患,在一些項(xiàng)目中 ,主函數(shù)循環(huán)中的任務(wù)越多,就意味著在某一瞬間,每顯示一位數(shù)碼管停留的時(shí)間就會(huì)越久,一旦超過某個(gè)值,會(huì)嚴(yán)重影響顯示的效果。這一節(jié)要教會(huì)大家兩個(gè)知識(shí)點(diǎn):

本文引用地址:http://m.butianyuan.cn/article/201611/319781.htm

第一個(gè):如何把動(dòng)態(tài)掃描數(shù)碼管的程序放在定時(shí)中斷里,徹底解決上節(jié)的顯示隱患。

第二個(gè):在定時(shí)中斷里的重裝初始值不能太大,否則動(dòng)態(tài)掃描數(shù)碼管的速度就不夠。我把原來常用的初始值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ù)碼管

* 通過設(shè)置其公共位com為高電平來關(guān)閉顯示,只要切換畫面的速度足夠快,人的視覺就分辨不出來,感覺八個(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í)的方式來延時(shí),目的是在八位數(shù)碼管中切換到每位數(shù)碼管顯示的時(shí)候,都能停留一會(huì)再切換到其它

* 位的數(shù)碼管界面,這樣可以增加顯示的效果。但是,由于朱兆祺51學(xué)習(xí)板是間接經(jīng)過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的輸出引腳上并且鎖存起來

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的輸出引腳上并且鎖存起來

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ù)碼管的速度就不夠。我把原來常用的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í)行一條空語句

}

}

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í)行一條空語句

}

}

}

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)過另外兩個(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ù),

* 了解來龍去脈,就可以知道本驅(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ì)疑,很多教科書上說,定時(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)芯片來實(shí)現(xiàn)靜態(tài)驅(qū)動(dòng)。因?yàn)閯?dòng)態(tài)掃描數(shù)碼管本來就不適合應(yīng)用在實(shí)時(shí)性非常高的項(xiàng)目。

前面這兩節(jié)都講了數(shù)碼管的驅(qū)動(dòng)程序,要在此基礎(chǔ)上,做一些項(xiàng)目中經(jīng)常遇到的界面應(yīng)用,我們?cè)撛趺磳懗绦?欲知詳情,請(qǐng)聽下回分解-----數(shù)碼管通過切換窗口來設(shè)置參數(shù)。



評(píng)論


技術(shù)專區(qū)

關(guān)閉