lcd1602计算器.docx
《lcd1602计算器.docx》由会员分享,可在线阅读,更多相关《lcd1602计算器.docx(14页珍藏版)》请在冰豆网上搜索。
lcd1602计算器
1.实验目的。
1、掌握单片机定时器、中断、键盘扫描、动态显示及数值计算的综合运用能力;
2、掌握编程的运用能力;
3、设计一只简易的16键计算器。
2.实验内容。
图5-1简易16键计算器
编程实现图5-1的简易16键4位十进制整数计算器的功能。
要求:
1.设计4*4的键盘,其中10个数字键0~9,其余6个为“+”“-”“*”“/”“=”和“C”
2.设计4位LED接口电路或LCD显示电路
3.实现2位数的简单运算
编程提示:
1、定时显示
开辟一显示缓冲区,显示中断程序定时读取缓冲区显示各位数字,每位显示2.5ms,显示4位需要10ms,也就是显示的刷新频率是100Hz。
这样既保证亮度又不闪烁,同时CPU还有足够时间做其他事情。
2、键盘中断扫描
利用键盘中断扫描程序,读出按键的键码,存入键盘缓冲区,供主程序读取。
3、计算器的主程序读取键盘缓冲区的键码,驱动计算器的运行。
4、计算中需要十进制与十六进制(或二进制)转化。
3、系统硬件设计电路图
矩阵键盘的硬件接线仿真图
LCD1602的接线仿真图
P0.0~P0.7引脚接lcd的D0-D7,P3口接矩阵键盘。
单片机的P2.4、P2.0、P2.3脚分别接到LCD的RS、R/W、E管。
整体电路图为:
显示电路,LCD1602A的引脚详细介绍:
采用LCD1602A来显示,要求能显示两行。
它的
—
引脚分别接到单片机的P0.0~P0.7引脚;RS、R/W、E管脚分别接到单片机的P2.4、P2.0、P2.3。
具体连接见后面的电路图。
以下是1602的16进制ASCII码表
4、软件设计。
包含程序主框图,所有程序。
分析可知共分为七个模块:
键盘扫描部分,定义各个位置上的键值部分,延迟部分,存贮部分,液晶显示部分,运算部分,动态显示部分。
运用了proteus软件仿真电路,使用keil软件进行程序的编写与调试。
Keil软件编写过程截图:
Proteus软件仿真电路图编写过程截图:
主程序流程图如下:
代码:
#include
#defineuintunsignedint
#defineucharunsignedchar
sbitlcden=P2^3;
sbitrs=P2^4;
sbitrw=P2^0;
sbitbusy=P0^7;
chari,j,temp,num,num_1;
longa,b,c;
floata_c,b_c;
ucharflag,fuhao;
ucharcodetable[]={
7,8,9,0,
4,5,6,0,
1,2,3,0,
0,0,0,0};
ucharcodetable1[]={
7,8,9,0x2f-0x30,
4,5,6,0x2a-0x30,
1,2,3,0x2d-0x30,
0x01-0x30,0,0x3d-0x30,0x2b-0x30};
voiddelay(ucharz)
{
uchary;
for(z;z>0;z--)
for(y=0;y<110;y++);
}
voidcheck()
{
do{
P0=0xFF;
rs=0;
rw=1;
lcden=0;
delay
(1);
lcden=1;
}while(busy==1);
}
voidwrite_com(ucharcom)
{
P0=com;
rs=0;
rw=0;
lcden=0;
check();
lcden=1;
}
voidwrite_date(uchardate)
{
P0=date;
rs=1;
rw=0;
lcden=0;
check();
lcden=1;
}
voidinit()
{
num=-1;
lcden=1;
write_com(0x01);
write_com(0x38);
write_com(0x0c);
write_com(0x06);
write_com(0x80);
write_com(0x01);
num_1=0;
i=0;
j=0;
a=0;
b=0;
c=0;
flag=0;
fuhao=0;
}
voidkeyscan()
{
P1=0xfe;
if(P1!
=0xfe)
{
delay(20);
if(P1!
=0xfe)
{
temp=P1&0xf0;
switch(temp)
{
case0xe0:
num=0;
break;
case0xd0:
num=1;
break;
case0xb0:
num=2;
break;
case0x70:
num=3;
break;
}
}
while(P1!
=0xfe);
if(num==0||num==1||num==2)
{
if(j!
=0)
{
write_com(0x01);
j=0;
}
if(flag==0)
{
a=a*10+table[num];
}
else
{
b=b*10+table[num];
}
}
else
{
flag=1;
fuhao=4;
}
i=table1[num];
write_date(0x30+i);
}
P1=0xfd;
if(P1!
=0xfd)
{
delay(5);
if(P1!
=0xfd)
{
temp=P1&0xf0;
switch(temp)
{
case0xe0:
num=4;
break;
case0xd0:
num=5;
break;
case0xb0:
num=6;
break;
case0x70:
num=7;
break;
}
}
while(P1!
=0xfd);
if(num==4||num==5||num==6&&num!
=7)
{
if(j!
=0)
{
write_com(0x01);
j=0;
}
if(flag==0)
{
a=a*10+table[num];
}
else
{
b=b*10+table[num];
}
}
else
{
flag=1;
fuhao=3;
}
i=table1[num];
write_date(0x30+i);
}
P1=0xfb;
if(P1!
=0xfb)
{
delay(5);
if(P1!
=0xfb)
{
temp=P1&0xf0;
switch(temp)
{
case0xe0:
num=8;
break;
case0xd0:
num=9;
break;
case0xb0:
num=10;
break;
case0x70:
num=11;
break;
}
}
while(P1!
=0xfb);
if(num==8||num==9||num==10)
{
if(j!
=0)
{
write_com(0x01);
j=0;
}
if(flag==0)
{
a=a*10+table[num];
}
else
{
b=b*10+table[num];
}
}
elseif(num==11)
{
flag=1;
fuhao=2;
}
i=table1[num];
write_date(0x30+i);
}
P1=0xf7;
if(P1!
=0xf7)
{
delay(5);
if(P1!
=0xf7)
{
temp=P1&0xf0;
switch(temp)
{
case0xe0:
num=12;
break;
case0xd0:
num=13;
break;
case0xb0:
num=14;
break;
case0x70:
num=15;
break;
}
}
while(P1!
=0xf7);
switch(num)
{
case12:
{write_com(0x01);a=0;b=0;flag=0;fuhao=0;}
break;
case13:
{
if(flag==0)
{
a=a*10;
write_date(0x30);
P1=0;
}
elseif(flag==1)
{
b=b*10;
write_date(0x30);
}
}
break;
case14:
{j=1;
if(fuhao==1){write_com(0x80+0x4f);
write_com(0x04);
c=a+b;
while(c!
=0)
{
write_date(0x30+c%10);
c=c/10;
}
write_date(0x3d);
a=0;b=0;flag=0;fuhao=0;
}
elseif(fuhao==2){write_com(0x80+0x4f);
write_com(0x04);
if(a-b>0)
c=a-b;
else
c=b-a;
while(c!
=0)
{
write_date(0x30+c%10);
c=c/10;
}
if(a-b<0)
write_date(0x2d);
write_date(0x3d);
a=0;b=0;flag=0;fuhao=0;
}
elseif(fuhao==3){write_com(0x80+0x4f);
write_com(0x04);
c=a*b;
while(c!
=0)
{
write_date(0x30+c%10);
c=c/10;
}
write_date(0x3d);
a=0;b=0;flag=0;fuhao=0;
}
elseif(fuhao==4){write_com(0x80+0x4f);
write_com(0x04);
i=0;
c=(long)(((float)a/b)*1000);
while(c!
=0)
{
write_date(0x30+c%10);
c=c/10;
i++;
if(i==3)
write_date(0x2e);
}
if(a/b<=0)
write_date(0x30);
write_date(0x3d);
a=0;b=0;flag=0;fuhao=0;
}
}
break;
case15:
{write_date(0x30+table1[num]);flag=1;fuhao=1;}
break;
}
}
}
main()
{
init();
while
(1)
{
keyscan();
}
}
运行后的截图:
5、请写出实验过程中曾出现的问题和你的解决方法,你对实验有何感想和体会?
一开始,看到实验要求,可以使用八位数码管和lcd1602进行显示数据,我就想着使用lcd1602进行显示数据的了,因为虽然使用数码管显示比较简单,但是数码管一共就只有八位,除去运算符号和结果之外能进行的显示就只有一两位。
所以使用lcd602可以进行两行数据显示,这样看起来比较简单明了。
一开始我想着使用实物板子进行操作的,可是代码在仿真的时候可以,就是烧进板子之后就是不行,没按下一次按键就会把前面的一位数据给覆盖,而且控制的显示位置也不行,可是在仿真的时候都没出现过上述的问题,所以在尝试了两天之后我放弃使用实物板子进行试验,转而使用了仿真,可能失败的原因使自己对自己的板子不是经常的去勤加练习去熟悉板子。
我仿真的接线图与实物板子是大致相同的,所以我一开始就是按照仿造别人的代码进行修改从而使得仿真成功的。
而且运算的加减乘除都是可以成功的,而且结果也能成功,运算的位数比使用数码管显示数据还多,而且还能显示小数位。
虽然实物板子没有成功,但是仿真的结果还是很令我自己满意的,以后我会专心去研究这个在实物上面也会成功显示的。