1、优化上节《 51单片机——8位数码管显示正整数和小数及解决鬼影问题1.0》里的代码。
2、原来dt_display(xxx),显示一会就不显示了,本次直接优化掉了,数码管dt_display();放入main函数里即可。
3、增添数码管初始化函数。
4、去除了清空数码管函数,这个感觉没有啥用。
备注:转换小数convert_float()函数有点小问题,主要是小数精确度的问题,这个后期有眉目了在优化。
#ifndef __DT_H__
#define __DT_H__
#include "delay.h"
extern void RES_dt_dispaly(void);//复位初始化数码管,并显示8个0~8个9
//0≤x≤99999999 unsigned int 无符号整型
//将数据x转换到数组里面
extern void convert_long_int(unsigned long int x);
//x为8位有符号小数 for 对于DS18B20温度的,最小0.0000001℃,最大显示-999.9998℃ (浮点数运算精度丢失问题)
extern void convert_float(float i);//小数转换后存放到数组里面
//循环显示数组里的元素到数码管
extern void dt_display(void);
#endif
//数码管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(10);//亮的时间,防闪烁;怕影响定时器故不用10ms延迟了直接在定时器里50ms刷新一下..........
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 x)
{
unsigned char i = 0; //数组标号也是段选真值
unsigned char a = 0; //流水号,用来表示数码管1~8个
for (a = 0;a < 8;a++) //将x没位的值存放到数组里面
{
i = x % 10;//获取最后一位数字
if ((a!=0)&&(x==0))//防止x=0时,导致数码管不显示0
{
i = 24;//即0x00,数码管不会亮
}
x = x / 10;//每次舍去最低位
datax[a] = dt_duan[i];
}
}
//x为8位有符号小数 for 对于DS18B20温度的,最小0.0000℃,最大显示-255.9999℃ (浮点数运算精度丢失问题)
void convert_float(float i)//小数转换后存放到数组里面
{
bit b; //b用来存放这个i是正小数还是负小数
unsigned char c; //用来存放i的正整数部分
unsigned int d; //用来存放i的小数部分
unsigned char e; //流水号
unsigned char ii; //数组标号
unsigned char a; //辅助变量,判断小数部分几位数
if (i < 0)
{
b = 1;//表示为负小数
i = -i;//将这个负小数转化为正小数
}
else
{
b = 0;//表示为正小数
}
c = (unsigned char)i; //存放整数部分0<c<255
i =i-c+0.00005;//浮点数精度差
d = (unsigned int)(i * 10000);//只保留小数点后4位,在后面的直接舍去
/* if (d)//移除小数部分放大部分后面多余的0
{
a = 4;
do
{
e = d % 10;
if (!e)
{
d /= 10;
a--;
}
} while (!e);//移除小数部分放大部分后面多余的0
}
*/
// for (e = 0; e < a; e++)//保存小数部分
for (e = 0; e < 4; e++)//保存小数部分
{
ii = d % 10;//获取最后一位数字
d = d / 10;//每次舍去最低位
datax[e] = dt_duan[ii];
}
for (e ; e < 8; e++)//保存整数部分
{
ii = c % 10;//获取最后一位数字
//if (e == a)
if (e == 4)
{
datax[e] = dt_duan[ii] + dt_duan[10];//让此位数字带小数点进行显示
}
else if (c==0)//c=0时,数字已经显示完了,数字最前面的0就不需要显示了
{
if (b)//用来显示负号,在此处当c=0表示已经显示完了该显示负号了
{
datax[e] = dt_duan[11];//用来显示负号
b = 0;//清除负号标识为,防止显示多个
}
else
{
datax[e] = dt_duan[24];//即0x00,数码管不会亮
}
}
else if(c!=0)
{
datax[e] = dt_duan[ii];//c即不等于0,也不是显示小数点的哪个,故要正常显示了
}
c = c / 10;//每次舍去最低位
}
}
void RES_dt_dispaly(void)//清除数码管,关闭数码管,初始化数码管
{
unsigned char i;
unsigned long int y=11111111;
unsigned long int z = 1000000;
dt_dt = dt_wei[8];//清除位码,关闭所有数码管
wei = 1;
wei = 0;
duan = 1;
duan = 0;//恢复单片机IO口默认高电平状态
for (i = 0; i < 8; i++)
{
convert_long_int(y);
while (z--)
{
dt_display();
}
y += 11111111;
z = 1000000;
}
dt_dt = dt_wei[8];//清除位码,关闭所有数码管
wei = 1;
wei = 0;
duan = 1;
duan = 0;//恢复单片机IO口默认高电平状态
}