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口默认高电平状态
}