使用74hc573鎖存器的8位數(shù)碼管顯示函數(shù)
仿真效果截圖:
部分源代碼:
/************************************************\
* 8位共陽數(shù)碼管顯示函數(shù)(使用74hc573鎖存器) *
* 力求程序高效,簡潔 *
\************************************************/
/************************************************\
* 分多個函數(shù)的原因: *
* 顯示要求不同,函數(shù)的執(zhí)行效率不同 *
* 為了實(shí)現(xiàn)程序的高效性,顯示要求不同調(diào)用不同的函數(shù)*
\************************************************/
#include
#define DISPLAY_IO P2
#define uchar unsigned char
#define uint unsigned int
#define ulong unsigned long
sbit duan=P3^0; //定義段
sbit wei=P3^1; //定義位
unsigned char code w[]={ //共陰
//0-F無小數(shù)點(diǎn)
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71,
//0-F帶小數(shù)點(diǎn)
0xbf,0x86,0xdb,0xcf,
0xe6,0xed,0xfd,0x87,
0xff,0xef,0xf7,0xfc,
0xb9,0xde,0xf9,0xf1,
//小數(shù)點(diǎn) 負(fù)號
0x80, 0x40};
/************************************************\
* 顯示任意內(nèi)容 *
* n[0-7]數(shù)組的值表示數(shù)碼管位置從左至右的內(nèi)容 *
\************************************************/
void display_any(uchar n[8])
{
uchar i,delay=0;
for(i=0;i<8;i++)
{
//清除段鎖存器內(nèi)容
DISPLAY_IO=0xff;
duan=1;
duan=0;
//位值送入位鎖存器
DISPLAY_IO=0x01<
wei=1;
wei=0;
//段值送入段鎖存器
DISPLAY_IO=0xff-n[i];
duan=1;
duan=0;
//延時0-255(越長越亮,但太長會閃)
delay=100;
while(delay--);
}
//清除段鎖存器內(nèi)容 不讓第一個數(shù)最亮
DISPLAY_IO=0xff;
duan=1;
duan=0;
}
/************************************************\
* 10進(jìn)制無符號整數(shù)顯示 *
* 傳入n為0到9999 9999 *
* 傳入浮點(diǎn)型小數(shù)點(diǎn)后忽略,范圍不正確顯示不正確 *
* 右對齊,左邊無多余0 *
\************************************************/
void display_ulong(ulong n)
{
uchar i=8,delay=0;
do
{
//清除段鎖存器內(nèi)容
DISPLAY_IO=0xff;
duan=1;
duan=0;
//位值送入位鎖存器
DISPLAY_IO=0x01<<(--i);
wei=1;
wei=0;
//段值送入段鎖存器
DISPLAY_IO=0xff-w[n%10];
duan=1;
duan=0;
n/=10;
//延時0-255(越長越亮,但太長會閃)
delay=100;
while(delay--);
}while(n>0);
//清除段鎖存器內(nèi)容 不讓第一個數(shù)最亮
DISPLAY_IO=0xff;
duan=1;
duan=0;
}
/************************************************\
* 10進(jìn)制有符號整數(shù)顯示 *
* 傳入n為-999 9999到9999 9999 *
* 傳入浮點(diǎn)型小數(shù)點(diǎn)后忽略,范圍不正確顯示不正確 *
* 右對齊,左邊無多余0 *
\************************************************/
void display_long(long n)
{
uchar i=8,delay=0;
bit flag=0;
if(n<0)
{
flag=1;
n=-n;
}
do
{
//清除段鎖存器內(nèi)容
DISPLAY_IO=0xff;
duan=1;
duan=0;
//位值送入位鎖存器
DISPLAY_IO=0x01<<(--i);
wei=1;
wei=0;
//段值送入段鎖存器
DISPLAY_IO=0xff-w[(n==100000000?33:n%10)];
duan=1;
duan=0;
n/=10;
if(n==10000000)
n=-1;
if(n==0&&flag==1)
n=100000000;
//延時0-255(越長越亮,但太長會閃)
delay=100;
while(delay--);
}while(n>0);
//清除段鎖存器內(nèi)容 不讓第一個數(shù)最亮
DISPLAY_IO=0xff;
duan=1;
duan=0;
}
/************************************************\
* 10進(jìn)制無符號實(shí)型顯示 *
* 傳入n為0到9999 9800 *
* 傳入范圍不正確顯示不正確 *
* 右對齊,左邊無多余0,右邊小數(shù)點(diǎn)后無多余0 *
\************************************************/
void display_udouble(double n)
{
uchar i=8,delay=0,point_position=0;
ulong n_long=n; //將n賦值給整形變量
while(n_long!=n) //此循環(huán)的功能:使n變?yōu)?位整數(shù),并記下小數(shù)點(diǎn)位置
{
n*=10;
n_long=n;
point_position++;//每乘一次,小數(shù)點(diǎn)左移一位
}
do
{
//清除段鎖存器內(nèi)容
DISPLAY_IO=0xff;
duan=1;
duan=0;
//位值送入位鎖存器
DISPLAY_IO=0x01<<(--i);
wei=1;
wei=0;
//段值送入段鎖存器
DISPLAY_IO=0xff-w[n_long%10+((7-i)==point_position?16:0)];
duan=1;
duan=0;
n_long/=10;
//延時0-255(越長越亮,但太長會閃)
delay=100;
while(delay--);
}while(n_long>0);
//清除段鎖存器內(nèi)容 不讓第一個數(shù)最亮
DISPLAY_IO=0xff;
duan=1;
duan=0;
}
/************************************************\
* 10進(jìn)制有符號實(shí)型顯示 *
* 傳入n為-999 999到9999 9800 *
* 傳入范圍不正確顯示不正確 *
* 右對齊,左邊無多余0,右邊小數(shù)點(diǎn)后無多余0 *
\************************************************/
void display_double(double n)
{
uchar i=8,delay=0,point_position=0;
bit flag=0;
long n_long=n; //將n賦值給整形變量
while(n_long!=n) //此循環(huán)的功能:使n變?yōu)?位整數(shù),并記下小數(shù)點(diǎn)位置
{
n*=10;
n_long=n;
point_position++;//每乘一次,小數(shù)點(diǎn)左移一位
}
if(n_long<0)
{
flag=1;
n_long=-n_long;
}
do
{
//清除段鎖存器內(nèi)容
DISPLAY_IO=0xff;
duan=1;
duan=0;
//位值送入位鎖存器
DISPLAY_IO=0x01<<(--i);
wei=1;
wei=0;
//段值送入段鎖存器
DISPLAY_IO=0xff-w[(n_long==100000000?33:(n_long%10+((7-i)==point_position?16:0)))];
duan=1;
duan=0;
n_long/=10;
if(n_long==10000000)
n_long=-1;
if(n_long==0&&flag==1)
n_long=100000000;
//延時0-255(越長越亮,但太長會閃)
delay=100;
while(delay--);
}while(n_long>0);
//清除段鎖存器內(nèi)容 不讓第一個數(shù)最亮
DISPLAY_IO=0xff;
duan=1;
duan=0;
}
/************************************************\
* 2-16任意進(jìn)制無符號不去零整數(shù)顯示 *
* 傳入n為0到X,X=((進(jìn)制)^8)-1 *
* 傳入范圍不正確顯示不正確 *
* 不自動去0 *
\************************************************/
void display_number(ulong n,uchar jin_zhi)
{
uchar i,delay=0;
for(i=0;i<8;i++)
{
//清除段鎖存器內(nèi)容
DISPLAY_IO=0xff;
duan=1;
duan=0;
//位值送入位鎖存器
DISPLAY_IO=0x01<<(7-i);
wei=1;
wei=0;
//段值送入段鎖存器
DISPLAY_IO=0xff-w[n%jin_zhi];
duan=1;
duan=0;
n/=jin_zhi;
//延時0-255(越長越亮,但太長會閃)
delay=100;
while(delay--);
}
//清除段鎖存器內(nèi)容 不讓第一個數(shù)最亮
DISPLAY_IO=0xff;
duan=1;
duan=0;
}
/************************\
* 8位數(shù)碼管測試程序 *
\************************/
void main(void)
{
uint i,t=200;
char n[8]={1,2,4,8,16,32,64,128};
while(1)
{
i=t;
while(i--)
{
display_any(n);
}
i=t;
while(i--)
{
display_ulong(123);
}
i=t;
while(i--)
{
display_long(-123);
}
i=t;
while(i--)
{
display_udouble(123.45);
}
i=t;
while(i--)
{
display_double(-123.45);
}
i=t;
while(i--)
{
display_number(0x123abc,16);
}
}
}