动态数码管的位选通过下面的38译码器来控制。通过1,2,3接口控制Y0-Y7的输出。将8个输出与上图的8个动态数码管的阴极接在一起。
下面是38译码器的真值表:从如果A2->A0为高地址到低位地址来看,则刚好3为2进制,转换为10进制数对应输入Y1->Y7的值为0.
下面是完整代码:
#include <reg51.h>
typedef unsigned int u16;
typedef unsigned char u8;
sbit LSA=P2^2;
sbit LSB=P2^3;
sbit LSC=P2^4;
void delay1s(void) //延迟1s,误差 0us
{
unsigned char a,b,c;
for(c=167;c>0;c–)
for(b=171;b>0;b–)
for(a=16;a>0;a–);
}
void delay(u16 i)
{
while(i–);//i=1,约为10微秒
}
char code num[17]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e,0xff};
//数码管 0-f 16个数,加0xff不显示
//加个code 指令将num这个数组放到code段内就是rom段。
void DigDisplay()
{
u8 i;
for(i=0;i<8;i++)//位选
{
switch(i)
{
case 0:
LSA=0;
LSB=0;
LSC=0;
//P0=~num[i];//设置段选
break;
case 1:
LSA=1;
LSB=0;
LSC=0;
//P0=~num[i];//设置段选
break;
case 2:
LSA=0;
LSB=1;
LSC=0;
//P0=~num[i];//设置段选
break;
case 3:
LSA=1;
LSB=1;
LSC=0;
//P0=~num[i];//设置段选
break;
case 4:
LSA=0;
LSB=0;
LSC=1;
//P0=~num[i];//设置段选
break;
case 5:
LSA=1;
LSB=0;
LSC=1;
//P0=~num[i];//设置段选
break;
case 6:
LSA=0;
LSB=1;
LSC=1;
//P0=~num[i];//设置段选
break;
case 7:
LSA=1;
LSB=1;
LSC=1;
//P0=~num[i];//设置段选
break;/**/
}
P0=~num[i+2];//设置段选
delay(100);
P0=0;
}
}
void main()
{
//P0=6;//共阴数码管, 1
//P0=7;//
P0=0xff;//共阳数码管, 2
while(1)
{
DigDisplay();
}
}