在上一篇中,我们已经说过了基于51单片机的简单拨号器,在下边,我们将写一个计算器程序,原理很简单,只需要在拨号器的基础上,算出拨号器所表示的数字,并进行计算即可。
代码如下;
#include"reg51.h"
#include typedef unsigned char u8; //对数据类型进行声明定义 typedef unsigned int u16; sbit LSA=P2^2; //38译码器数码管位选 sbit LSB=P2^3; sbit LSC=P2^4; sbit k1=P3^1; sbit k2=P3^0; sbit k3=P3^2; sbit k4=P3^3; #define GPIO_KEY P1 #define GPIO_DIG P0 u16 KeyValue; //用来存放读取到的键值 u16 keyflag,i; //用来回复是否有按键按下 u8 code smgduan[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07, 0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71}; //显示0~F的值 u16 wei[8]={0}; //用来存放每一位数码管数字的数组 /******************************************************************************* * 函 数 名 : delay * 函数功能 : 延时函数,i=1时,大约延时10us *******************************************************************************/ void delay(u16 i) { while(i--); } /******************************************************************************* * 函 数 名 : display * 函数功能 : 扫描显示数码管 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ void display() { LSA=0; LSB=0; LSC=0; P0=smgduan[wei[0]];delay(50); P0=0x00; LSA=1; LSB=0; LSC=0; P0=smgduan[wei[1]];delay(50); P0=0x00; LSA=0; LSB=1; LSC=0; P0=smgduan[wei[2]];delay(50); P0=0x00; LSA=1; LSB=1; LSC=0; P0=smgduan[wei[3]];delay(50); P0=0x00; LSA=0; LSB=0; LSC=1; P0=smgduan[wei[4]];delay(50); P0=0x00; LSA=1; LSB=0; LSC=1; P0=smgduan[wei[5]];delay(50); P0=0x00; LSA=0; LSB=1; LSC=1; P0=smgduan[wei[6]];delay(50); P0=0x00; LSA=1; LSB=1; LSC=1; P0=smgduan[wei[7]];delay(50); P0=0x00; } /******************************************************************************* * 函 数 名 : KeyDown * 函数功能 : 检测有按键按下并读取键值 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ void KeyDown(void) { char a=0; GPIO_KEY=0x0f; if(GPIO_KEY!=0x0f)//读取按键是否按下 { keyflag=1; delay(1000);//延时10ms进行消抖 if(GPIO_KEY!=0x0f)//再次检测键盘是否按下 { //测试列 GPIO_KEY=0X0F; switch(GPIO_KEY) { case(0X07): KeyValue=0;break; case(0X0b): KeyValue=1;break; case(0X0d): KeyValue=2;break; case(0X0e): KeyValue=3;break; } //测试行 GPIO_KEY=0XF0; switch(GPIO_KEY) { case(0X70): KeyValue=KeyValue;break; case(0Xb0): KeyValue=KeyValue+4;break; case(0Xd0): KeyValue=KeyValue+8;break; case(0Xe0): KeyValue=KeyValue+12;break; } while((a<50)&&(GPIO_KEY!=0xf0)) //检测按键松手检测 { delay(1000); a++; } } } } /******************************************************************************* * 函 数 名 : main * 函数功能 : 主函数 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ void main() { u16 a=0,b=0,c=0; while(1) { display(); /* 第一个数字输入*/ KeyDown(); if(keyflag==1) { for(i=7;i>0;i--) //输入一位,数字向左移动一位 {wei[i]=wei[i-1];} wei[0]=KeyValue; keyflag=0; } /*************************加法运算****************************/ if(k1==0) { a=wei[0]+wei[1]*10+wei[2]*100+wei[3]*1000+wei[4]*10000+wei[5]*100000+wei[6]*1000000+wei[7]*10000000; //计算a的值 for(i=0;i<8;i++) wei[i]=0; //对数码管清零 while(1) //输入第二个数 { display(); KeyDown(); if(KeyValue==15) break; if(keyflag==1) { for(i=7;i>0;i--) {wei[i]=wei[i-1];} wei[0]=KeyValue; keyflag=0; } } b=wei[0]+wei[1]*10+wei[2]*100+wei[3]*1000+wei[4]*10000+wei[5]*100000+wei[6]*1000000+wei[7]*10000000; //计算b的值 c=a+b; wei[0]=c%10; //计算C的各个位的数字 wei[1]=c/10%10; wei[2]=c/100%10; wei[3]=c/1000%10; wei[4]=c/10000%10; wei[5]=c/100000%10; wei[6]=c/1000000%10; wei[7]=c/10000000%10; while(1) { display(); } } /*************************减法运算****************************/ else if(k2==0) { a=wei[0]+wei[1]*10+wei[2]*100+wei[3]*1000+wei[4]*10000+wei[5]*100000+wei[6]*1000000+wei[7]*10000000; //计算a的值 for(i=0;i<8;i++) wei[i]=0; //对数码管清零 while(1) { //输入第二个数 display(); KeyDown(); if(KeyValue==15) break; //当读到等于号,既,KeyValue=15时,停止输入