51单片机——8位数码管显示正整数和小数及解决鬼影问题1.2

1、优化上节《 51单片机——8位数码管显示正整数和小数及解决鬼影问题1.1》里的代码。

2、原来的小数对于DS18B20温度转换有问题,本次直接解决了,对DS18B20温度转换想了解的请看本人的《51单片机——DS18B20温度传感器-1总线通讯1.2

3、对DS18B20转换问题(小数、正负号小数、正整数)修改了原来的convert_long_int()这个函数。本次主要看void convert_long_int(unsigned long int DATA,bit flag_bit,unsigned char Accuracy_bit);和void dt_convert_DS18B20(void);

备注:原转换小数convert_float()函数有点小问题,主要是小数精确度的问题,这个本期已更正。

感觉写的好就关注一下哈。

dt.h

#ifndef __DT_H__
#define	__DT_H__

#include "delay.h"
#include "ds18b20.h"

extern void RES_dt_dispaly(void);//复位初始化数码管,并显示8个0~8个9

//-9999999≤x≤99999999 flag_bit 正数为0,负数为1,Accuracy_bit小数个数
//将数据x转换到数组里面
void convert_long_int(unsigned long int x,bit flag_bit,unsigned char Accuracy_bit);

//小数转换后存放到数组里面
void dt_convert_DS18B20(void);

//循环显示数组里的元素到数码管
extern void dt_display(void);

#endif

dt.c

//数码管dt  使用共阴极数码管即位选是阴极时数码管才选中
//使用P0-8个GPIO口
//P2.7位选脚(控制哪个数码管亮)、P2.6段选脚(送数码管值) 控制2个74锁存器

#include <reg52.h>
#include "dt.h"

#define dt_dt  P0
sbit wei = P2^7;		//锁存器位选
sbit duan = P2^6;		//锁存器段选

static unsigned char datax[8];//定义局部变量//用来存放小数及正整数,然后发给数码管进行显示低位到高位存放
static code unsigned char dt_duan[25] =   //数码管段选真值表   dt_duan[i] + 0x80  数码管显示数据并且此数据的右下角有小数点
{
	0x3F,  //"0"
	0x06,  //"1"
	0x5B,  //"2"
	0x4F,  //"3"
	0x66,  //"4"
	0x6D,  //"5"
	0x7D,  //"6"
	0x07,  //"7"
	0x7F,  //"8"
	0x6F,  //"9"
	0x80,  //"."
	0x40,  //"-"
	0x77,  //"A"
	0x7C,  //"B"
	0x39,  //"C"
	0x5E,  //"D"
	0x79,  //"E"
	0x71,  //"F"
	0x76,  //"H"
	0x38,  //"L"
	0x37,  //"n"
	0x3E,  //"u"
	0x73,  //"P"
	0x5C,  //"o"
	0x00  //熄灭
};

static code unsigned char dt_wei[9]={0x7f,0xbf,0xdf,0xef,0xf7,0xfb,0xfd,0xfe,0xff};//1位-8位(0x7f表示最右边哪个数码管)0xff数码管就不亮了

//刷新数码管进行显示
void dt_display(void)//循环显示数组里的元素到数码管
{
	unsigned char a=0;
	for (a = 0; a < 8; a++)//a亮——a值——a值灭;a+1亮——a+1值——a+1值灭;开始循环(亮,真值,假值)
	{
		dt_dt = dt_wei[a];//灯亮
		wei = 1;            //打开位选
		wei = 0;            //锁存位选
			
		dt_dt = datax[a];//数字
		duan = 1;           //打开段选
		duan = 0;           //锁存段选
			
		delay_ms(1);//这个时间乘以8不能大于定时器时间
			
		dt_dt = dt_duan[24];//清楚段码数字熄灭,假值,防止下位数码管显示上次的值
		duan = 1;           //打开段选
		duan = 0;           //锁存段选
	}
	dt_dt = dt_wei[8];//清除位码,关闭所有数码管
}

//将要显示的正整数临时存储在数组里面,然后循环显示数组里的元素即可
//x传递的数据10进制(x不是8位则高位不会亮)
void convert_long_int(unsigned long int DATA,bit flag_bit,unsigned char Accuracy_bit)
{
	unsigned char value;		//计算结果值
	unsigned char a;		//流水号,用来表示数码管1~8个
	unsigned char b;//流水号
	switch(Accuracy_bit)//判断精度
	{
		case 1:b=1;break;
		case 2:b=2;break;
		case 3:b=3;break;
		case 4:b=4;break;
		case 5:b=5;break;
		case 6:b=6;break;
		case 7:b=7;break;//如果高位大于10则1和负号则失真了整体个数不能大于8位(不含小数点)
		default:b=0;break;
	}
	for (a = 0;a < b;a++)//Accuracy_bit的值存放到数组里面
	{
		value = DATA % 10;//获取最后一位数字
		datax[a] = dt_duan[value];
		DATA = DATA / 10;//每次舍去最低位
	}
	for(;a<8;a++)
	{
		value=DATA % 10;//获取最后一位数字
		if(a==b)//要加入小数学
		{
			datax[a] = dt_duan[value]+dt_duan[10];
		}
		else if(DATA==0)
		{
			if(flag_bit)
			{
				flag_bit=0;
				datax[a]=dt_duan[11];//显示-号
			}
			else
				datax[a]=dt_duan[24];//不显示0
		}
		else
		{
			datax[a] = dt_duan[value];
		}
		DATA = DATA / 10;//每次舍去最低位
	}
}

//对于DS18B20温度的
void dt_convert_DS18B20(void)//小数转换后存放到数组里面
{
	static unsigned char value;		//计算结果值
	static unsigned char a;		//流水号,用来表示数码管1~8个
	static unsigned char b;//流水号
	switch(DS18B20_Accuracy_bit)//判断精度
	{
		case 9:b=1;break;
		case 10:b=2;break;
		case 11:b=3;break;
		default:b=4;break;
	}
	for (a = 0;a < b;a++)//Accuracy_bit的值存放到数组里面
	{
		value = DS18B20_int_bit % 10;//获取最后一位数字
		datax[a] = dt_duan[value];
		DS18B20_int_bit = DS18B20_int_bit / 10;//每次舍去最低位
	}
	for(;a<8;a++)
	{
		value=DS18B20_int_bit % 10;//获取最后一位数字
		if(a==b)//要加入小数学
		{
			datax[a] = dt_duan[value]+dt_duan[10];
		}
		else if(DS18B20_int_bit==0)
		{
			if(DS18B20_flag_bit)
			{
				DS18B20_flag_bit=0;
				datax[a]=dt_duan[11];//显示-号
			}
			else
				datax[a]=dt_duan[24];//不显示0
		}
		else
		{
			datax[a] = dt_duan[value];
		}
		DS18B20_int_bit = DS18B20_int_bit / 10;//每次舍去最低位
	}
}

void RES_dt_dispaly(void)//清除数码管,关闭数码管,初始化数码管
{
	unsigned char i;
	unsigned long int y=11111111;
	unsigned long int z = 63;//大概1秒
	dt_dt = dt_wei[8];//清除位码,关闭所有数码管
	wei = 1;
	wei = 0;

	duan = 1;
	duan = 0;//恢复单片机IO口默认高电平状态
	for (i = 0; i < 8; i++)
	{
		convert_long_int(y,0,0);
		while (z--)
		{
			dt_display();
		}
		y += 11111111;
		z = 20;//大概很快
	}
	dt_dt = dt_wei[8];//清除位码,关闭所有数码管
	wei = 1;
	wei = 0;

	duan = 1;
	duan = 0;//恢复单片机IO口默认高电平状态
}

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
好的,这个问题解决方法比较简单。我们可以先使用dS18B20传感器读取当前的温度值,然后再将其显示数码管上。具体的步骤如下: 1. 首先,需要连接dS18B20传感器和数码管到单片机上。连接方法可以参考相关的硬件资料和引脚图。 2. 然后需要使用相应的dS18B20库函数读取当前的温度值。例如,如果使用基于STM32的开发板,可以使用HAL库提供的相关函数来读取。读取温度值的代码可能类似于以下代码: ```c float temperature = 0; uint8_t buffer[2]; HAL_Delay(1000); // 延时等待dS18B20转换完成 DS18B20_ReadData(buffer); // 读取dS18B20的温度值 temperature = buffer[0] + (float)buffer[1]/100; // 将温度值转换为浮点数 ``` 上面的代码中,我们首先使用HAL_Delay函数等待dS18B20传感器的转换完成。然后调用DS18B20_ReadData函数读取温度值,这个函数可能需要根据具体的硬件平台进行修改。最后,我们将读取到的温度值转换为浮点数,并存储在temperature变量中。 3. 最后,将读取到的温度值显示数码管上。这个步骤可以使用数码管驱动库函数来实现。例如,如果使用基于STM32的开发板,可以使用HAL库提供的GPIO和TIM相关函数来控制数码管显示。将温度值显示数码管上的代码可能类似于以下代码: ```c int integer_part = temperature; // 取出整数部分 int decimal_part = (temperature - integer_part)*100; // 取出小数部分 int decimal_first = decimal_part / 10; // 取出小数点后第一位数字 int decimal_second = decimal_part % 10; // 取出小数点后第二位数字 // 数码管显示整数部分 display_integer(integer_part); // 数码管显示小数部分 display_decimal(decimal_first, decimal_second); ``` 上面的代码中,我们首先将temperature变量的整数部分和小数部分分别取出。然后再将小数部分拆分为小数点后第一位数字和第二位数字。最后,将整数部分和小数部分分别显示数码管上。这里的display_integer和display_decimal函数可能需要根据具体的硬件平台进行修改。 综上所述,以上是将dS18B20读取的温度值显示数码管上的一个简单实现过程。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

金丝草

你的鼓励将是我创作的最大动力

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

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

打赏作者

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

抵扣说明:

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

余额充值