我想用51单片机,矩阵键盘,lcd1602写个简易计算器程序

如何把如float 类型的12.5转化成字符串格式以便在1602上显示12.5?求高手告知,最好能具体给个简单的例子。
c语言的谢谢

第1个回答  2011-08-05

//---------------------------------------------------

void W_LCD_X1_Dat(uint sec)

{

    W_LCD_Com(0xc0 + 6);

    W_LCD_Dat(0x30 + sec / 10000);

    W_LCD_Dat(0x30 + (sec % 10000) / 1000);

    W_LCD_Dat(0x30 + (sec % 1000) / 100);

    W_LCD_Dat(0x30 + (sec % 100) / 10);

    W_LCD_Dat('.');

    W_LCD_Dat(0x30 + sec % 10);

}

//---------------------------------------------------

void main()

{

    float aaa;

    uint  bbb;

    init();

    aaa = 12.5;

    bbb = aaa * 10;

    W_LCD_X1_Dat(bbb);

    while(1);

}

//========================================= 

初始化等函数,可见:

http://hi.baidu.com/%D7%F6%B6%F8%C2%DB%B5%C0/blog/item/e9a4f10963e835c463d98677.html ;

显示画面如下:

追问

呵呵,你这个是知道小数点后几位可以,如果位数未知就要先判断有几位,这样程序写起来很复杂

追答

在某个项目中,要求显示几位小数,应该是很明确的,不用判断。
通常,显示数字,四位就是很精密的了,常用是用三位显示。
那么,整数部分如果是0~9,仅仅需要显示2位小数;
整数部分如果是10~99,仅仅需要显示1位小数;
整数部分如果是100~999,就不显示小数。
可以说:真没有什么可判断的。
比使用字符转换函数什么的,要简单多的多了。

追问

哈哈,这个精度太差了。不适合计算器,倒是对有些ad转换后显示结果可以,比方说ds18b20
做的温度计可以

追答

你的题目要求仅仅是显示,怎么精度差?
内部计算,你愿意用几位,就用几位,没有精度差的问题。
单片机是用来作工控产品的,四位显示就够用。
用单片机做计算器,痴人说梦吧,这产品卖不出去的。

追问

嗯我做的不是产品只是练习一下程序,谢谢你了

追答

碰上循环小数,你还想要显示起来没有完吗?

应该显示多少位数,你自己规定即可。

如果你嫌精度差,你就多显示几位呗。

怎么可能是这个程序的精度差?

提问者,很明显,基本概念不清。

要是使用下面在步骤:
--把数字转化成字符放入数组,
--然后再通过调用函数转化成长整型,
--再把这个长整型可以用sprintf函数转化成字符

呵呵,真想像不出来,还有没有更笨的方法了。

花费单片机大量的时间,再多绕几个圈,也不会比本程序的精度高。

第2个回答  推荐于2016-04-25
C语言的语句可以这样写
#include<stdio.h>

void main()
{
float fxx;
unsigned char str[10];

fxx = 12.5;
sprintf(str,"%8.2f\n",fxx); //将12.5转换成8.2(2位小数)格式以ASCII方式存储到str字符串中
..........
}

由于1620液晶模块支持ASCII码,所以只要将str中的内容送到1602就行了。至于1602的操作我就不写了(其实我也没用过,但知道怎么用,应该不难)追问

这样好像不行,因为sprintf直接转数字如111,跟本转不成“111”

追答

怎么会呢?我仿真在str中看到的就是ASCII码,你的1602不支持吗?

追问

我用keil c软件做的。我当初就是卡在这个地方了,后来我发现这么做可以,就是先把输入的数字转化成字符放入数组,然后再通过调用函数转化成长整型,再把这个长整型可以用sprintf函数转化成字符没有错误

追答

我可能明白了,你对小数点的精度有要求是吧。单片机的C语言中如果使用标准数据类型,小数精度不可能很高(映像中好像是6、7位,其实也不算低了)。如果你想实现很高的精度,恐怕还是要采用多个long型的变量,自己编写算法(比如16位数的四则运算,很痛苦...)。输出处理上正如你所说,拼装成字符串后再输出。
关于使用sprintf,这是KEIL C自带的扩展函数,本身应该是没有问题的,输出不对可能与输出格式转换字符的写法有关。其实float型的很简单:
%f:x.xxx方式
%e:xxxxExx方式
%g:所谓的自动选择上述两种方式中更紧凑的一种作为输出格式。

希望对你有所借鉴。

本回答被提问者采纳
第3个回答  2011-08-06
#include <AT89X52.H>
#include <intrins.h>
#include "lcd_drive.h"
#define uchar unsigned char
#define uint unsigned int

void main()
{
uint key();
void deplay(uchar j_p);
void deplay_s(long a);

uchar i=1;
uchar jp=-1;
uchar flat;
uint a,b;

delay_ms(10);
lcd_init();
lcd_clr();

lcd_wcmd(0x80);

while(1)
{
while(jp==-1)
jp=key();

deplay(jp);

switch (jp)
{
case '=': switch(flat)
{
case '+': a+=b; deplay_s(a); break;
case '-': a=b-a; deplay_s(a); break;
case '*': a*=b; deplay_s(a); break;
case '/': a=b/a; deplay_s(a); break;
default:break;
}

break;
case '+':flat='+';b=a;i=0;break;
case '-':flat='-';b=a;i=0;break;
case '*':flat='*';b=a;i=0;break;
case '/':flat='/';b=a;i=0;break;
case 'c':a=0;b=0;lcd_clr();break;
case 0:
case 1:
case 2:
case 3:
case 4:
case 5:
case 6:
case 7:
case 8:
case 9:a=a*10*i+jp;i=1;break;
}

jp=-1;
}

}

//-------矩阵键盘扫描-------
uint key()
{
uchar h = 0,l = 0;//矩阵行h,列l
uchar j_p;

P3 = 0x0f;
if(P3 != 0x0f)
{
delay_ms( 2 );
if(P3!=0x0f)
{
l = P3 & 0x0f ;
P3 = 0xf0;
h = P3 & 0xf0;
}
}

switch(l | h)
{
case 0x7e:j_p=0;break; //按键0
case 0x7d:j_p=1;break; //按键1
case 0x7b:j_p=2;break; //按键2
case 0x77:j_p=3;break; //按键3
case 0xbe:j_p=4;break; //按键4
case 0xbd:j_p=5;break; //按键5
case 0xbb:j_p=6;break; //按键6
case 0xb7:j_p=7;break; //按键7
case 0xde:j_p=8;break; //按键8
case 0xdd:j_p=9;break; //按键9
case 0xdb:j_p='+';break; //按键a
case 0xd7:j_p='-';break; //按键b
case 0xee:j_p='=';break; //按键Enter
case 0xed:j_p='*';break; //按键c
case 0xeb:j_p='/';break; //按键e
case 0xe7:j_p='c';break; //按键取消
default:return -1;break;
}

P3=0x0f;
while(P3!=0x0f);

return j_p;
}

void deplay(uchar j_p)
{
if(j_p != -1)
{
if(j_p >= 0 && j_p <= 9)
j_p+=48;

lcd_wdat(j_p);
}
}

void deplay_s(long a)
{
int n[10];
uint i=0;
while(a)
{
n[i]=a%10;
a/=10;
i++;
}

while(i)
deplay(n[--i]);
}追问

你的程序我看了,算法倒是行,我想问一下,你这个算法在被除数无法被除数整除的时候结果是不是不能显示小数位?

相似回答