利用矩阵键盘实现一个简易的计算器。
为了简化问题。我们如果仅仅支持小于100的非负整数之间的加、减、乘的运算。而且支持连续运算(结果的数值能够再进行运算)。
本程序中C为加号,D为减号,E为乘号,F为等于号。
代码中有具体的凝视。
/* 注:本程序 C 为+, D 为- E为* F 为=号。支持非负整数连续运算。
输入的数值小于100,运算结果不超过1000. by Tach ------------------------------------------------*/ #include #define DataPort P0 //定义数据port 程序中遇到DataPort 则用P0 替换 #define KeyPort P3 sbit DUAN=P2^6;//定义锁存使能port 段锁存 sbit WEI=P2^7;// 位锁存 unsigned char code dofly_DuanMa[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f, 0x77,0x7c,0x39,0x5e,0x79,0x71,0x40};// 显示段码值0~F和-号 unsigned char code dofly_WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分别相应相应的数码管点亮,即位码 unsigned char TempData[8]; //存储显示值的全局变量 void DelayUs2x(unsigned char t);//us级延时函数声明 void DelayMs(unsigned char t); //ms级延时 void Display(unsigned char FirstBit,unsigned char Num);//数码管显示函数 unsigned char KeyScan(void);//键盘扫描 unsigned char KeyPro(void); void Init_Timer0(void);//定时器初始化 /*------------------------------------------------ 主函数 ------------------------------------------------*/ void main (void) { unsigned char num,tempp=0; int sym_add,sym_sub,sym_mul; int datanum[2]; int i=0,j,flag,ans,ans_clear,t; unsigned char temp[8]; Init_Timer0(); while (1) //主循环 { num=KeyPro(); if(num!=0xff) { if(num>=0 && num<=9) { if(ans_clear) { ans_clear=0; for(j=0;j<8;j++)//清屏 TempData[j]=0; } tempp=tempp*10+num; if(tempp/10>0) TempData[6]=dofly_DuanMa[tempp/10]; TempData[7]=dofly_DuanMa[tempp%10]; datanum[i]=tempp; } else if(num==15) { i=0; tempp=0; if(sym_add==1) { ans=datanum[0]+datanum[1]; t=ans; if(ans/100>0) { TempData[5]=dofly_DuanMa[ans/100]; ans=ans%100; } if(ans/10>0 || (TempData[5]!=0 && ans/10==0)) TempData[6]=dofly_DuanMa[ans/10]; TempData[7]=dofly_DuanMa[ans%10]; } else if(sym_sub==1) { ans=datanum[0]-datanum[1]; t=ans; if(ans<0) { flag=1; ans=-ans; } else flag=0; if(flag) TempData[4]=dofly_DuanMa[16]; //负号 if(ans/100>0) { TempData[5]=dofly_DuanMa[ans/100]; ans=ans%100; } if(ans/10>0) TempData[6]=dofly_DuanMa[ans/10]; TempData[7]=dofly_DuanMa[ans%10]; } else if(sym_mul==1) { ans=datanum[0]*datanum[1]; t=ans; if(ans/100>0) { TempData[5]=dofly_DuanMa[ans/100]; ans=ans%100; } if(ans/10>0 || (TempData[5]!=0 && ans/10==0)) TempData[6]=dofly_DuanMa[ans/10]; TempData[7]=dofly_DuanMa[ans%10]; } sym_add=0; sym_sub=0; sym_mul=0; ans_clear=1; datanum[0]=ans; } else if(num>=12 && num<=14) { i++; if(num==12) sym_add=1; else if(num==13) sym_sub=1; else if(num==14) sym_mul=1; tempp=0; for(j=0;j<8;j++)//清屏 TempData[j]=0; } } //主循环中加入其它须要一直工作的程序 } } /*------------------------------------------------ uS延时函数,含有输入參数 unsigned char t,无返回值 unsigned char 是定义无符号字符变量,其值的范围是 0~255 这里使用晶振12M,精确延时请使用汇编,大致延时 长度例如以下 T=tx2+5 uS ------------------------------------------------*/ void DelayUs2x(unsigned char t) { while(--t); } /*------------------------------------------------ mS延时函数,含有输入參数 unsigned char t,无返回值 unsigned char 是定义无符号字符变量,其值的范围是 0~255 这里使用晶振12M。精确延时请使用汇编 ------------------------------------------------*/ void DelayMs(unsigned char t) { while(t--) { //大致延时1mS DelayUs2x(245); DelayUs2x(245); } } /*------------------------------------------------ 显示函数。用于动态扫描数码管 输入參数 FirstBit 表示须要显示的第一位,如赋值2表示从第三个数码管開始显示 如输入0表示从第一个显示。
Num表示须要显示的位数,如须要显示99两位数值则该值输入2 ------------------------------------------------*/ void Display(unsigned char FirstBit,unsigned char Num) { static unsigned char i=0; DataPort=0; //清空数据,防止有交替重影 DUAN=1; //段锁存 DUAN=0; DataPort=dofly_WeiMa[i+FirstBit]; //取位码 WEI=1; //位锁存 WEI=0; DataPort=TempData[i]; //取显示数据,段码 DUAN=1; //段锁存 DUAN=0; i++; if(i==Num) i=0; } /*------------------------------------------------ 定时器初始化子程序 ------------------------------------------------*/ void Init_Timer0(void) { TMOD |= 0x01; //使用模式1,16位定时器,使用"|"符号能够在使用多个定时器时不受影响 //TH0=0x00; //给定初值 //TL0=0x00; EA=1; //总中断打开 ET0=1; //定时器中断打开 TR0=1; //定时器开关打开 } /*------------------------------------------------ 定时器中断子程序 ------------------------------------------------*/ void Timer0_isr(void) interrupt 1 { TH0=(65536-2000)/256; //又一次赋值 2ms TL0=(65536-2000)%256; Display(0,8); // 调用数码管扫描 } /*------------------------------------------------ 按键扫描函数,返回扫描键值 ------------------------------------------------*/ unsigned char KeyScan(void) //键盘扫描函数,使用行列逐级扫描法 { unsigned char Val; KeyPort=0xf0;//高四位置高,低四位拉低 if(KeyPort!=0xf0)//表示有按键按下 { DelayMs(10); //去抖 if(KeyPort!=0xf0) { //表示有按键按下 KeyPort=0xfe; //检測第一行 if(KeyPort!=0xfe) { Val=KeyPort&0xf0; Val+=0x0e; while(KeyPort!=0xfe); DelayMs(10); //去抖 while(KeyPort!=0xfe); return Val; } KeyPort=0xfd; //检測第二行 if(KeyPort!=0xfd) { Val=KeyPort&0xf0; Val+=0x0d; while(KeyPort!=0xfd); DelayMs(10); //去抖 while(KeyPort!=0xfd); return Val; } KeyPort=0xfb; //检測第三行 if(KeyPort!=0xfb) { Val=KeyPort&0xf0; Val+=0x0b; while(KeyPort!=0xfb); DelayMs(10); //去抖 while(KeyPort!=0xfb); return Val; } KeyPort=0xf7; //检測第四行 if(KeyPort!=0xf7) { Val=KeyPort&0xf0; Val+=0x07; while(KeyPort!=0xf7); DelayMs(10); //去抖 while(KeyPort!=0xf7); return Val; } } } return 0xff; } /*------------------------------------------------ 按键值处理函数,返回扫键值 ------------------------------------------------*/ unsigned char KeyPro(void) { switch(KeyScan()) { case 0xee:return 0;break;//0 按下相应的键显示相相应的码值 case 0xde:return 1;break;//1 case 0xbe:return 2;break;//2 case 0x7e:return 3;break;//3 case 0xed:return 4;break;//4 case 0xdd:return 5;break;//5 case 0xbd:return 6;break;//6 case 0x7d:return 7;break;//7 case 0xeb:return 8;break;//8 case 0xdb:return 9;break;//9 case 0xbb:return 10;break;//a case 0x7b:return 11;break;//b case 0xe7:return 12;break;//c case 0xd7:return 13;break;//d case 0xb7:return 14;break;//e case 0x77:return 15;break;//f default:return 0xff;break; } }