Proteus8+Keil5仿真 用AT89C51实现共阴极四位数码管显示数字0-9999

文章讲述了作者在编写代码时遇到数码管快速闪烁的问题,通过调整i值和取数方式,增加了循环次数并添加延迟,解决了多位数字同时闪烁的问题,展示了优化过程和代码片段。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最开始本来是想着让 i 循环 10000 次,然后通过
dp[i % 10]
dp[i / 10 % 10]
dp[i / 100 % 10]
dp[i / 1000 % 10]
来取到每位数码管上的要展示的数字,但是写完代码发现数码管上的数字变化的特变快,然后就想着在最后加一个delay(500)进行延迟但结果是四位数字一起闪烁(就像快速的眨眼睛)。消影的延迟又不能动于是就想到了让i扩大同时扩大取每位数的方式使得循环总次数增加,经过不断调试找到了比较好的实现方式

代码如下:

#include <reg51.h>
#include <intrins.h> 

#define uint unsigned int
#define uchar unsigned char

//定义0-9的16进制
uchar code dp[10] = {
  0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x07, 0x7f, 0x6f
};

// 延迟函数
void delay(uchar ms) {
   uchar i, j;
    for (i = 0; i < ms; ++i) {
        for (j = 0; j < 120; ++j);
    }
}


void main() {
	uint i = 0;
	while(1){
			  P1 = 0xfe;
				P2 = dp[i / 20 % 10];
				delay(5);
				P1= 0xff;
			
			  P1 = 0xfd;
				P2 = dp[(i / 200) % 10];
				delay(5);
				P1= 0xff;
			
			  P1 = 0xfb;
				P2 = dp[(i / 2000) % 10];
				delay(5);
				P1= 0xff;
			
			  P1 = 0xf7;
				P2 = dp[(i / 20000) % 10];
				delay(5);
				P1= 0xff;
		
		delay(100);
			if(i <=  200000) i++;
	}
}

仿真电路图:

<think>嗯,用户想要用ProteusKeil通过AT89C51单片机控制数码管静态显示0到9循环。首先,我需要理清楚整个项目的步骤,确保电路设计和代码部分都正确。 首先,硬件部分。数码管的话,每个段的阳极是连在一起的,接到VCC,而阴极则需要通过单片机控制低电平来点亮对应的段。所以数码管的a到dp引脚应该连接到单片机的某个端口,比如P0口。记得要加上限流电阻,防止电流过大烧坏LED。通常每个段可能需要220欧姆左右的电阻。然后,AT89C51需要连接时钟电路,比如12MHz晶振,以及复位电路,简单的上电复位就可以。这部分在Proteus里应该不难画,主要注意数码管的类型是否正确选择为阳。 接下来是软件部分。代码需要让P0口依次输出对应的段码,然后循环延时。数码管的段码和阴的是相反的,所以0到9的编码需要确认是否正确。例如,数字0的段码应该是0xC0,因为阳的话,各段除了g和dp都点亮,对应的二进制是11000000,即0xC0。然后每个数字对应的段码需要列出来,组成一个数组。循环遍历这个数组,每次输出到P0,然后延时一段时间,比如500毫秒,这样就能看到数字变化了。 不过用户提到的是静态显示,所以可能不需要动态扫描,直接每次改变P0口的输出即可。这时候可能需要检查数码管的阳公端是否正确连接,比如是否接VCC,或者是否需要通过三极管驱动,如果电流不够的话,但静态显示单个的话,可能直接接VCC也可以,但具体要看数码管的规格。 然后,在Keil里编写代码时,需要包含AT89X51的头文件,配置好项目,选择正确的单片机型号,生成HEX文件。然后在Proteus里加载这个HEX文件,进行仿真。延时函数可能需要用传统的循环方式,因为不使用定时器中断的话,简单延时可以用for循环实现,不过精度不高,但仿真应该没问题。 另外,要注意Proteus中的元件名称是否正确,比如数码管是否是7SEG-COM-ANODE,或者类似的型号。连接P0口时,是否需要上拉电阻,因为P0口在51单片机内部是开漏的,所以可能需要外接上拉电阻,否则无法输出高电平。但这里因为是数码管,段码输出低电平时点亮,所以P0口作为输出低电平的话,可能不需要上拉,但为了稳定,可能还是加上比较好。或者,用户可能没有接上拉,导致仿真时无法正常工作,这点需要提醒。 代码部分,主函数里循环遍历段码数组,然后通过P0输出,延时。可能需要用unsigned char类型定义变量和数组。还有,循环变量是否溢出,比如从0到9,然后回到0,所以用i = (i+1)%10;这样的方式。 可能出现的问题包括段码错误,导致显示数字不正确,或者延时太短或太长,看不到变化。另外,Proteus仿真中是否加载了正确的HEX文件,单片机时钟设置是否正确,比如如果代码里用的是12MHz,而Proteus中晶振也是12MHz,这样延时才会准确。 总结一下,步骤应该是:设计电路图(单片机数码管、电阻、晶振、复位电路),编写C51程序,生成HEX,仿真测试。需要确保硬件连接正确,代码逻辑正确,特别是段码和延时部分。</think>根据您的要求,我将分步骤为您讲解基于ProteusKeil数码0-9循环显示实现方案: 一、Proteus电路图设计要点 1. 核心器件清单: - AT89C51 - 7SEG-COM-ANODE(数码管) - 220Ω电阻排(RESPACK-8- 12MHz晶振 - 30pF电容×2 - 10kΩ电阻 - 10μF电解电容 2. 电路连接示意图: ```plaintext P0.0-P0.7 → 数码管a-dp段(通过220Ω限流电阻) P2.0数码管公端(阳接法) XTAL1/2 → 12MHz晶振+30pF电容 RST → 上电复位电路 ``` 二、C51程序代码(Keil工程) ```c #include <reg51.h> #define uint unsigned int #define uchar unsigned char // 数码管段码表(0-9) uchar code SEG_CODE[] = { 0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8, 0x80, 0x90}; void delay(uint t) { uint i, j; for(i = t; i > 0; i--) for(j = 110; j > 0; j--); } void main() { uchar i = 0; P2 = 0x01; // 公端使能 while(1) { P0 = SEG_CODE[i]; delay(500); i = (i + 1) % 10; } } ``` 三、关键实现原理 1. 数码管驱动: - 数码管采用灌电流驱动方式 - P0口输出低电平点亮对应段码 - 使用电阻排限制段电流(约10mA/段) 2. 定时控制: - 通过双重循环实现500ms延时 - 修改delay参数可调整显示速度 四、注意事项 1. 硬件验证: - 确认数码管公端已接VCC - 检查P0口是否需要上拉电阻(建议添加1kΩ排阻) 2. 软件调试: -Keil中设置正确的晶振频率 - 生成HEX文件后需正确加载到Proteus单片机中 3. 扩展改进建议: - 添加按键控制启/停 - 增加动态扫描支持多位数码- 采用定时器中断实现精确计时 实际仿真时,数码管应每秒变化一次显示数字,循环显示0-9。如需调整显示速度,可修改delay函数的参数值。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值