标题:
单片机计算器源程序(矩阵键盘+数码管显示)原理图与Proteus仿真 代码注释很详细
[打印本页]
作者:
51黑电子迷
时间:
2017-5-7 21:46
标题:
单片机计算器源程序(矩阵键盘+数码管显示)原理图与Proteus仿真 代码注释很详细
矩阵键盘+数码管显示组成的一个简易计算器单片机项目
下面是单片机计算器的proteus仿真原理图(工程文件可到本帖附件中下载):
0.png
(31.7 KB, 下载次数: 112)
下载附件
2017-5-7 21:43 上传
单片机计算器源程序如下:
//#include <reg51.h>
# include <STC12C5A60S2.h>
# define uchar unsigned char
# define uint32 unsigned long int
# define sint32 signed long int
# define vtime 3000 //定时3ms,一帧8*3=24ms,频率=40Hz
//共阴数码管段码表
uchar code distable[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0X00,0x40,0x31};
//位选码表
uchar code numi[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};
//显示缓存
uchar V_ram[]={0x10,0x10,0x10,0x10,0x10,0x10,0x10,0x10};
uchar error=0,wi=0,dot1=0,dot2=0,dot=0xff; //位选循环变量
uint32 num1=0,num2=0,num=0,sum=0;
void dispsum();
void delay(uchar);
uchar scan_key();
main() //m1:m0 00=标准; 01=推挽; 10=输入; 11=开漏输出
{
uchar ni,nj,fn='+',key_V=0,dotp=0,j,k,i=0;
uint32 temp=0;
//m1:m0 00=标准; 01=推挽; 10=输入; 11=开漏输出
P1M1 = 0X0f;
P1M0 = 0XF0; //设定低4位为输入,高4位为输出
P0M1 = 0X00;
P0M0 = 0Xff;
P2M1 = 0X00;
P2M0 = 0Xff; //设定P0,P2推挽输出
TMOD = 0X01; //设定定时器0为16位计数方式
TH0 = (65536-vtime )/256;
TL0 = (65536-vtime )%256; //赋定时器0初值
ET0 = 1; //开定时器0中断
EA = 1; //开总中断
TR0 = 1; //启动定时器0计数
//sum=123456;
while(1)
{
key_V=scan_key();
//dispsum();
if(key_V)
{
//有键按下
key_V--;
switch(key_V)
{
case 0:
case 1:
case 2:
case 3:
case 4:
case 5:
case 6:
case 7:
case 8:
case 9:
if((num+i)==0)
{ //如果是第一个数字输入,则清黑屏
dot=0xff; //关闭小数点
for(j=0;j<8;j++)
V_ram[j]=0x10; //清黑屏
}
num=num*10+key_V; //键盘输入的数字转化为十进制的数值
if(i==8) //如果显示缓冲满了
{
i=7;
for(j=0;j<7;j++)
V_ram[j]=V_ram[j+1];//显示缓存串行移位
}
V_ram[i]=key_V; //键值放入显示缓存
i++;
break;
case 10: //dot 小数点符号
dotp=i; //记录下小数点位置
dot=dotp-1; //显示小数点
break;
case 11: //+ 加法符号
for(j=0;j<8;j++)
V_ram[j]=0x10; //清黑屏
num1=num; //获取从键盘输入的第一个数值
if(dotp==0)
{
dot1=0;
}
else
{
dot1=i-dotp;
}
ni=i;
dotp=0;
dot=0xff;
num=0;
fn='+';
i=0;
break;
case 12: //- 减法符号
for(j=0;j<8;j++)
V_ram[j]=0x10; //清黑屏
num1=num; //获取从键盘输入的第一个数值
if(dotp==0)
{
dot1=0;
}
else
{
dot1=i-dotp;
}
ni=i;
dotp=0;
dot=0xff;
num=0;
fn='-';
i=0;
break;
case 13: //* 乘法符号
for(j=0;j<8;j++)
V_ram[j]=0x10; //清黑屏
num1=num; //获取从键盘输入的第一个数值
if(dotp==0)
{
dot1=0;
}
else
{
dot1=i-dotp;
}
ni=i;
dotp=0;
dot=0xff;
num=0;
fn='*';
i=0;
break;
case 14: // / 除法符号
for(j=0;j<8;j++)
V_ram[j]=0x10; //清黑屏
num1=num; //获取从键盘输入的第一个数值
if(dotp==0)
{
dot1=0;
}
else
{
dot1=i-dotp;
}
ni=i;
dotp=0;
dot=0xff;
num=0;
fn='/';
i=0;
break;
case 15: //= 等于符号
if(dotp==0) //计算第二个运算数的小数位数
{
dot2=0;
}
else
{
dot2=i-dotp;
}
dotp=0;
nj=i;
temp=1;
if(fn!='*')
{ if(dot1>dot2) //第一个运算数与第二个运算进行对齐
{
for(j=dot1-dot2;j>0;j--)
{
//if(num>0x19999999)error=1;
num=num*10;
nj++;
if(nj>9)
{
j--;
for(k=j;k>0;k--)
num1=num1/10;
dot1=dot1-j;
break;
}
}
}
else
{
for(j=dot2-dot1;j>0;j--)
{
//if(num1>0x19999999)error=1;
num1=num1*10;
ni++;
if(ni>9)
{
j--;
for(k=j;k>0;k--)
num=num/10;
dot2=dot2-j;
break;
}
}
}
}
//准备好后,进行运算
if(fn=='+') //做加法运算
{
sum=num+num1;
if(sum<(num|num1))error=1;
dispsum(); //显示对齐后的运算结果
if(dot2>dot1) //显示实际小数点的位置
dot=dot-dot2;
else
dot=dot-dot1;
i=0; //计算结束,初始化关键变量
num1=0;
num=0;
break;
}
if(fn=='-') //做减法运算
{
if(num1<num) //判断结果是负还是正
{
sum=num-num1;
}
else
sum=num1-num;
dispsum(); //显示对齐后的运算结果
if(num>num1) //是负数则第一位显示“-”,其它位顺序往后显示
{
for(j=7;j>0;j--)
V_ram[j]=V_ram[j-1]; //顺序往后显示
V_ram[0]=0x11; //第一位显示“-”
dot++; //小数点也顺延一位
}
if(dot2>dot1) //显示实际小数点的位置
dot=dot-dot2;
else
dot=dot-dot1;
i=0; //计算结束,初始化关键变量
num1=0;
num=0;
break;
}
if(fn=='*') //做乘法运算
{
sum=num1*num;
if((ni+nj)>10)error=1;
dispsum(); //显示对齐后的运算结果
//if(dot2>dot1) //显示实际小数点的位置
dot=dot-dot2-dot1;
//else
// dot=dot-dot1-dot1;
i=0; //计算结束,初始化关键变量
num1=0;
num=0;
break;
}
if(fn=='/') //做除法运算
{
sum=num1/num;
dispsum(); //显示对齐后的运算结果,对于除法,就是实际结果
temp=num1%num; //显示小数部分
for(j=dot;j<7;j++)
{
temp=temp*10;
V_ram[j+1]=temp/num;
temp=temp%num;
if(temp==0)break;
}
i=0; //计算结束,初始化关键变量
num1=0;
num=0;
break;
}
break;
default:
break;
}
delay(200);//延时去抖
}
}
}
void t0_isp() interrupt 1
{
uchar dm,wx;
TH0 = (65536-vtime )/256;
TL0 = (65536-vtime )%256; //赋定时器0初值
dm=distable[V_ram[wi]]; //取显示段码
wx=numi[wi]; //取位选码
P2=0x00; //关显示
//P0=dm; //段码赋给P0口
if(dot==wi)
{
P0=dm|0x80; //点亮位选的那个数码管+小数点
}
else
{
P0=dm; //点亮位选的那个数码管
}
P2=wx;
wi++;
if(wi==8)wi=0;
}
/////////////////////////////////////
uchar scan_key()
{
uchar i,j,vlume=0,temp[4]={0xef,0xdf,0xbf,0x7f};
for(j=0;j<4;j++)
{
P1=temp[j];
i=P1|0xf0;
if(i != 0xff) //判断按键
{ //有键按下
i=~i;
if(i==4)i=3;
if(i==8)i=4;
i--; // 计算按键所在行号(0--3)
vlume=i*4+j+1; //计算键值
//delay(200);//延时去抖
}
}
return vlume;
}
////////////////////////////////////////
void delay(uchar k )
{
uchar x,y,z;
for(x=k;x>0;x--)
for(y=20;y>0;y--)
for(z=250;z>0;z--);
}
////////////////////////////////////////
void dispsum()
{
uchar temp[10]=0,k,j;
uint32 temp1;
if(error==1)
{
V_ram[0]=0x0e;
V_ram[1]=0x12;
for(j=2;j<8;j++)
V_ram[j]=0x10; //清黑屏
dot=0xff;
error=0;
num=0;
return;
}
temp1=sum;
for(k=0;k<10;k++)
{
temp[k]=temp1%10; // sum的个位开始放入temp数组,最多可放10位
temp1=temp1/10; //
if(temp1==0)
{ // 如果商为零
dot=k; //当前位小数点点亮
for(j=0;j<8;j++)
{
if(k==0xff)
{
V_ram[j]=0x10; // 清黑屏
}
else
{
V_ram[j]=temp[k]; // 把sum中的数从高位依次放入显示缓存
k--;
…………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码
最后给大家分享一些我们老师给的一些经典的单片机程序源码, 一共有十多个.都有详细的注释,然大家快速的理解每一行代码的意思。而且有proteus仿真原理图。大家可以直接验证程序的对错.
本系列所有源码打包下载地址(含proteus仿真工程文件和源程序):
http://www.51hei.com/bbs/dpj-82474-1.html
本例程下载:
计算器.rar
(100.85 KB, 下载次数: 89)
2017-5-7 21:43 上传
点击文件名下载附件
下载积分: 黑币 -5
作者:
295102889
时间:
2019-5-2 21:27
感谢分享 好东西啊
欢迎光临 (http://www.51hei.com/bbs/)
Powered by Discuz! X3.1