静态数码管显示
在使用数码管之前,我们首先要对数码管的工作原理有一个了解,请查看板子的数码管模块原理图
可以看到,一共有8位数字,每位数字由8个小LED构成(a、b、c、d、e、f、g、dp)每个数字的对应LED都连在一起,并且与74HC245相连,74HC245的作用是进行缓冲,当DIR端口接高电平时固定为A0传输给B0,A1给B1...
为什么不直接相连而要进行缓冲呢?
因为如果直接相连的话,高电平从寄存器P0传来的信号已经很小,很难再驱动这么多信号,所以此时借用DIR端的高电平,只需要检测到P0传来的对应的信号就可以使用芯片使Vcc的电信号传输给8个LED端口,因此寄存器P0就可以实现操控LED显示的数字。例如:当想要显示数字8时,只需要使a、b、c、d、e、f、g都为亮,也就是接通高电平,此时P0应赋值为0111 1111即0x7F,一位数赋值情况如下表:
0:0011 1111即0x3F | 1:0000 0110即0x06 |
---|---|
2:0101 1011即0x5B | 3:0100 1111即0x4F |
4:0110 0110即0x66 | 5:0110 1101即0x6D |
6:0111 1101即0x7D | 7:0000 0111即0x07 |
8:0111 1111即0x7F | 9:0110 1111即0x6F |
如何选择哪一位来显示数字呢?通过查看端口LED0~7,发现他们都与译码器74HC138相连通过寄存器P2中的第2到4为进行选择,当P24为1,当P23为0,当P24为0时,选中Y4即LED5进行显示,此时LED5端口为低电平,所以可以解释为什么P0中的值为1时小LED亮。
举例实现一个使LED2显示6的代码:
#include <REGX52.H> void main() { //想要使LED2亮,就要选中Y1,所以使P2的第四位到第二位分别为001(二进制高位对应寄存器高位) P2_2=1; P2_3=0; P2_4=0; P0=0x7D;//显示6,dp和b端为0不亮,其余为1亮 对应0111 1101 while(1){} }
这样使用数码管未免有些过于麻烦,为了方便后续的数码管使用,我们可以将LED显示封装为一个函数,只需要给这个函数两个信息,一个是显示的位置,一个是显示的值,这样就可以便捷的使用数码管。实现如下:
#include <REGX52.H> unsigned char Nixienum[]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F}; //将要显示的每一个数的八个小LED熄灭情况转换为十六进制表示,并与数组下标对应起来 void Nixie(unsigned char Location,num) { switch(Location)//根据传递过来的Location来选择对应位置的LED显示 { case 1:P2_4=1;P2_3=1;P2_2=1;break; case 2:P2_4=1;P2_3=1;P2_2=0;break; case 3:P2_4=1;P2_3=0;P2_2=1;break; case 4:P2_4=1;P2_3=0;P2_2=0;break; case 5:P2_4=0;P2_3=1;P2_2=1;break; case 6:P2_4=0;P2_3=1;P2_2=0;break; case 7:P2_4=0;P2_3=0;P2_2=1;break; case 8:P2_4=0;P2_3=0;P2_2=0;break; } P0=Nixienum[num];//显示对应数字 } void main() { Nixie(2,6); //只需要传递一个位置和想要显示的数即可 while(1) { } }
动态数码管显示
动态数码管显示就是多个LED同时显示不同的数字,但是我们的数码管显示函数只能实现一位数字的显示,但是,左脚踩右脚,一定能上天。只要我们显示的够快,每位数显示的时候,只需要在它消失之前,再次显示一遍,它就能永远留在上面。
因此我们把想要显示的不同位置的不同的数字放入循环体内,使它一直不断地刷新,就可以实现不同的数字同时留在上面。
但是此时我们会发现,数字居然有些错位了!为什么会这样?
举个例子,当我上前一位数的LED灯的a是亮的,但是下一位的a是灭的,我还没来得及熄,你就已经换了下一位LED,所以就会留下一点残影。
如何消除残影呢?
只需要在函数后面加一个延迟一毫秒的Delay函数,这样每次显示完一位成后都会停止一会儿,这样就可以消除残影了。
总体实现如下:
#include <REGX52.H> unsigned char Nixienum[]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F}; void Delay(int n) //@12.000MHz ms { unsigned char data i, j; while(n--) { i = 2; j = 239; do { while (--j); } while (--i); } } void Nixie(unsigned char Location,num) { switch(Location) { case 1:P2_4=1;P2_3=1;P2_2=1;break; case 2:P2_4=1;P2_3=1;P2_2=0;break; case 3:P2_4=1;P2_3=0;P2_2=1;break; case 4:P2_4=1;P2_3=0;P2_2=0;break; case 5:P2_4=0;P2_3=1;P2_2=1;break; case 6:P2_4=0;P2_3=1;P2_2=0;break; case 7:P2_4=0;P2_3=0;P2_2=1;break; case 8:P2_4=0;P2_3=0;P2_2=0;break; } P0=Nixienum[num]; Delay(1); //消影 } void main() { while(1) { Nixie(1,1); Nixie(2,2); Nixie(3,3); Nixie(4,4); } }