首先打开proteus,导入8位数码管和89c51。
然后如图连线,分清断码和位码, 断码就是一个数码管的7个LED灯。
位码:就是第几位显示,由于是共阴极,所以哪位接地就显示哪位。
下面通过改变位码的接线就可以看出不同的效果
下面就编写程序,从第1位到第8位显示从0到7的八个数字。也就是位码的第一位为低电平,然后断码的除了g灯不亮其他都亮也就是0x3f,这样第一位就显示了0,后面就依次类推。
下面写c程序,之后编译。可复制后直接享用!
#include"reg51.h"
unsigned char dat[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
unsigned char wei[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};
// 点 g f e d c b a
// 显示0 0 1 1 1 1 1 1 除了点和g不亮其他的灯都亮就是0。 十六进制就是0x3f
// 显示1 0 0 0 0 1 1 0 只有b和c亮,其他的都不亮。 十六进制就是0x06
// 按照上面的方式依次类推得到 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f
void delay(unsigned int n){ //延时函数 不是很精确
unsigned int i=0, j=0;
for(i=0; i<n; i++){
for(j=0; j<120; j++);
}
}
void seg(){ // 自己命名的子函数,proteus中数码管叫seg,我这里就取名叫seg好了
unsigned int i;
for(i=0; i<8; i++){ // 循环10次,得到i的值从0到7都能取到
P3=~wei[i]; // 位码从第0位到第7位
P2=dat[i]; // 断码从0到7显示
delay(5); // 延时不能太长了,太长了人眼就能捕捉到有灯灭掉了,就不是持续点亮的状态了。
}
}
void main(){ // 主函数 加一个无限循环 只调用了子函数seg
while(1){
seg();
}
}
把编译的hex文件导入后就可以看到0到7都显示出来了,而且也不闪动,如果有闪动请降低延时数。
由于这样接线太占用IO端口了,所以我们选用一个译码器来减少IO口的使用,用3个口控制8个口,这个译码器叫74ls138。
如上图所示,这个芯片本省就带取反功能,也就是Y0输出1的条件是输入000 P端口十六进制为0x00
Y1输出1的条件是输入001 P端口十六进制为0x01
Y2输出1的条件是输入010 P端口十六进制为0x02
Y3输出1的条件是输入011 P端口十六进制为0x03
Y4输出1的条件是输入100 P端口十六进制为0x04
Y5输出1的条件是输入101 P端口十六进制为0x05
Y6输出1的条件是输入110 P端口十六进制为0x06
Y7输出1的条件是输入111 P端口十六进制为0x07
根据这样的条件就可以用3个IO口控制数码管的8位位码了,节省5个IO口。如下图所示:
下面编写c程序并编译。
#include"reg51.h"
unsigned char dat[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
unsigned char wei[]={0x00,0x01,0x02,0x03,0x04,0x05,0x06,0x07};
// 点 g f e d c b a
// 显示0 0 1 1 1 1 1 1 除了点和g不亮其他的灯都亮就是0。 十六进制就是0x3f
// 显示1 0 0 0 0 1 1 0 只有b和c亮,其他的都不亮。 十六进制就是0x06
// 按照上面的方式依次类推得到 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f
void delay(unsigned int n){ //延时函数 不是很精确
unsigned int i=0, j=0;
for(i=0; i<n; i++){
for(j=0; j<120; j++);
}
}
void seg(){ // 自己命名的子函数,proteus中数码管叫seg,我这里就取名叫seg好了
unsigned int i;
for(i=0; i<8; i++){ // 循环10次,得到i的值从0到7都能取到
P3=wei[i]; // 位码从第0位到第7位输出给译码器,同样也能控制第0位到第七位。
P2=dat[i]; // 断码从0到7显示
delay(5); // 延时不能太长了,太长了人眼就能捕捉到有灯灭掉了,就不是持续点亮的状态了。
}
}
void main(){ // 主函数 加一个无限循环 只调用了子函数seg
while(1){
seg();
}
}
proteus导入编译好的hex文件后模拟运行,就可以看到数码管从0到7显示出来了。
好了,这就是本文的主要内容。
接下来就是用led屏做一个实时时钟,效果如下图:
用proteus导入相关硬件,并接线。接下来就是写0-9的LED显示的16进制值了,我是在本子上写出来的,然后抄到代码上的。主要是记录过程:
接下来就是关于中断的一些知识点,也是要了解的:
上面的知识点都了解了,就是写程序了,直接上代码,变量都是汉语拼音,复制粘贴就可以直接使用了:
#include<reg52.h>
unsigned int led[10]={0x40, 0xf9, 0xa4, 0xb0, 0x99, 0x92, 0x02, 0xf8, 0x80, 0x90};
unsigned char heng=0xBF;
unsigned int count=0;
unsigned char xiaoshi=15, fenzhong=53, miaozhong=53; // 这里可以直接修改时分秒的赋值,从而改变时间
void InitTime()
{
TMOD=0x01;
TH0=0x4c;
TL0=0x00;
EA=1;
TR0=1;
ET0=1;
}
void delay(unsigned int t)
{
unsigned int i=0,j=0;
for(i=0;i<t;i++)
{
for(j=0;j<120;j++);
}
}
void display()
{
unsigned char xs=xiaoshi/10, xg=xiaoshi%10, fs=fenzhong/10, fg=fenzhong%10, ms=miaozhong/10, mg=miaozhong%10;
int i=0;
for(i=0;i<8;i++)
{
switch(i)
{
case 0:
{
P2=0x01;
P3=led[xs];
delay(1);
break;
}
case 1:
{
P2=0x02;
P3=led[xg];
delay(1);
break;
}
case 2: //从左到右第3位为-
{
P2=0x04;
P3=heng;
delay(1);
break;
}
case 3:
{
P2=0x08;
P3=led[fs];
delay(1);
break;
}
case 4:
{
P2=0x10;
P3=led[fg];
delay(1);
break;
}
case 5:
{
P2=0x20;
P3=heng;
delay(1);
break;
}
case 6:
{
P2=0x40;
P3=led[ms];
delay(1);
break;
}
case 7:
{
P2=0x80;
P3=led[mg];
delay(1);
break;
}
}
}
}
int main()
{
InitTime();
while(1)
{
display();
}
return 0;
}
void zhongduan() interrupt 1
{
TH0=0x4c;
TL0=0x00;
count++;
if(count == 20)
{
miaozhong++;
count=0;
}
if(miaozhong == 60)
{
fenzhong++;
miaozhong=0;
}
if(fenzhong == 60)
{
xiaoshi++;
fenzhong = 0;
}
if(xiaoshi == 24)
{
xiaoshi=0;
}
}
最后编译后倒入proteus中运行就可以看到效果了。
这里记录一下proteus中添加网络标号的快捷方法,要不每次鼠标右键效率太低了。
下面是快捷标线的方法,鼠标左键点击开始标注,省事快捷。文章来源:https://www.toymoban.com/news/detail-420780.html
文章来源地址https://www.toymoban.com/news/detail-420780.html
到了这里,关于51单片机控制数码管动态显示的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!