基於51微控制器的簡單計算器
阿新 • • 發佈:2018-12-30
在上一篇中,我們已經說過了基於51微控制器的簡單撥號器,在下邊,我們將寫一個計算器程式,原理很簡單,只需要在撥號器的基礎上,算出撥號器所表示的數字,並進行計算即可。
程式碼如下;
#include"reg51.h"
#include<intrins.h>
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時,停止輸入
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(k3==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時,停止輸入
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(k4==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時,停止輸入
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(); //顯示最終結果
}
}
}
}