51单片机控制数码管动态显示

首先打开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中添加网络标号的快捷方法,要不每次鼠标右键效率太低了。

下面是快捷标线的方法,鼠标左键点击开始标注,省事快捷。

  • 27
    点赞
  • 235
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

尚久龙

你的鼓励是我最大的动力!谢谢!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值