51单片机——计算器编写的一种办法(C语言版)

计算器功能包括:

        1、支持8位数内(0~99999999)的包括加、减、乘、除的非负整数运算。

        2、软件清零功能。 

        3、通过串口通信输入功能。

        4、八位共阴极数码管显示。

注:

        不具有断电后储存的功能,参与计算的数字一旦超出规定,如果是运算数按前8位记录。

在程序实现上核心有两个模块:显示和键扫

程序实现:

显示:

       通过取模获得每位数,然后通过查库(数列)获得对应的段码和位码,再依次扫描显示到数码管上。显示程序如下:

#include <Delay.h>//调用延时函数
#define DataPort P1//定义数码管数据来源端口

sfr P1=0x90;
sfr P2=0xA0;
//定义P1和P2口
sbit LATCH1=P2^1;//定义段锁存
sbit LATCH2=P2^2;//定义位锁存

void Display(unsigned long int Number)
{
	unsigned int segment_code[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
	unsigned int bit_code[8]={0x7f,0xbf,0xdf,0xef,0xf7,0xfb,0xfd,0xfe};
    //设置段码和位码
	unsigned int x[8];//将数字分离出来的每一位
	unsigned int i,j;
	
	for(i=0;Number>=1;i++)
	{
		x[i]=Number%10;
		Number=(Number-x[i])/10;
	}
	j=i;

	for(i=0;i<j;i++)
	{
		DataPort=bit_code[i];
		LATCH2=1;
		LATCH2=0;
	
		DataPort=segment_code[x[i]];
		LATCH1=1;
		LATCH1=0;
	
		DelayMs(1);
	}
}

因为数码管是动态扫描的,肉眼会捕捉到数码管变化,类似动漫的原理,加上1ms左右延时即可在8位内“静态显示”。

延时函数如下(tms的延时效果):

void DelayUs2x(unsigned char t)
{
	while(--t);
}

void DelayMs(unsigned char t)
{
	while(t--)
	{
		DelayUs2x(245);
		DelayUs2x(245);
	}
}

调试样例如下:

void Display(unsigned long int Number);

main()
{
    Display(99999999);
}

调试结果如下:

键扫:

        本方法采用行列扫描的方法,键盘定义如下:

 键扫代码如下:

#define KeyPort P0//定义键盘端口

void Delay(unsigned int xms);//声明延时函数

sfr P0=0x80;//定义P0端口

sbit h_1=P0^0;
sbit h_2=P0^1;
sbit h_3=P0^2;
sbit h_4=P0^3;
sbit l_1=P0^4;
sbit l_2=P0^5;
sbit l_3=P0^6;
sbit l_4=P0^7;
//定义行列

unsigned char KeyScan()
{
	unsigned char KeyCode=0;
	
	KeyPort=0xff;
	l_4=0;
	if(h_1==0){Delay(20);while(h_1==0);Delay(20);KeyCode=1;}
	if(h_2==0){Delay(20);while(h_2==0);Delay(20);KeyCode=2;}
	if(h_3==0){Delay(20);while(h_3==0);Delay(20);KeyCode=3;}
	if(h_4==0){Delay(20);while(h_4==0);Delay(20);KeyCode=4;}
	
	KeyPort=0xff;
	l_3=0;
	if(h_1==0){Delay(20);while(h_1==0);Delay(20);KeyCode=5;}
	if(h_2==0){Delay(20);while(h_2==0);Delay(20);KeyCode=6;}
	if(h_3==0){Delay(20);while(h_3==0);Delay(20);KeyCode=7;}
	if(h_4==0){Delay(20);while(h_4==0);Delay(20);KeyCode=8;}
	
	KeyPort=0xff;
	l_2=0;
	if(h_1==0){Delay(20);while(h_1==0);Delay(20);KeyCode=9;}
	if(h_2==0){Delay(20);while(h_2==0);Delay(20);KeyCode=10;}
	if(h_3==0){Delay(20);while(h_3==0);Delay(20);KeyCode=11;}
	if(h_4==0){Delay(20);while(h_4==0);Delay(20);KeyCode=12;}
	
	KeyPort=0xff;
	l_1=0;
	if(h_1==0){Delay(20);while(h_1==0);Delay(20);KeyCode=13;}
	if(h_2==0){Delay(20);while(h_2==0);Delay(20);KeyCode=14;}
	if(h_3==0){Delay(20);while(h_3==0);Delay(20);KeyCode=15;}
	if(h_4==0){Delay(20);while(h_4==0);Delay(20);KeyCode=16;}
	
	return KeyCode;//返回码用于查表
}

void Delay(unsigned int xms)
{
	unsigned char i,j;
	while(xms--)
	{
		i=2;
		j=239;
		do
		{
			while(--j);
		}
		while(--i);
	}
}

 再通过查表的方法避免返回0的情况,代码如下:

unsigned char KeyTab(unsigned char Char)
{
	switch (Char)
	{
		case 1:return 7;break;
		case 2:return 4;break;
		case 3:return 1;break;
		case 4:return 0;break;
		case 5:return 8;break;
		case 6:return 5;break;
		case 7:return 2;break;
		case 8:return '~';break;
		case 9:return 9;break;
		case 10:return 6;break;
		case 11:return 3;break;
		case 12:return '=';break;
		case 13:return '+';break;
		case 14:return '-';break;
		case 15:return '*';break;
		case 16:return '/';break;
	}
}

计算程序如下:

unsigned long int Regulation(unsigned long int x,unsigned long int y,char  m)
{
	unsigned long int z;
	switch(m)
	{
		case '+':z=x+y;break;
		case '-':z=x-y;break;
		case '*':z=x*y;break;
		case '/':z=x/y;break;
	}
	return z;
}

综上结合后主函数:

#include <Function.h>

sfr P2=0xA0;

sbit Switch=P2^0;//另外定义的整个清零开关

main()
{
	unsigned char X,Y,M,p;
	unsigned long int A=0,B=0,Num;
	while(1)
	{
		if(!Switch)
		{
			p=0;
			A=0;
			B=0;
			Num=0;
		}
		X=KeyScan();
		if(X)
		{
			if(X>0&&X<12&&X!=8)
			{
				if(p==0)
				{
					Y=KeyTab(X);
					A=A*10+Y;
					Num=A;
				}
				else
				{
					Y=KeyTab(X);
					B=B*10+Y;
					Num=B;
				}
			}
			else if(X==12)
			{
				if(p==0)
				{
					Num=A;
				}
				else
				{
					Num=Regulation(A,B,M);
				}
			}
			else if(X==8)
			{
				if(p==0)
				{
					A=(A-Y)/10;
					Num=A;
				}
				else
				{
					B=(B-Y)/10;
					Num=B;
				}
			}
			else
			{
				M=KeyTab(X);
				p++;
			}
		}
		Display(Num);
	}
}

Function头文件:

#ifndef _FUNCTION_H_
#define _FUNCTION_H_

void Display(unsigned long int Number);

unsigned char KeyScan();

unsigned char KeyTab(unsigned char Char);

unsigned long int Regulation(unsigned long int x,unsigned long int y,char  m);

#endif

其中因为数组位数问题Display函数定义如下:

#define DataPort P1

sfr P1=0x90;
sfr P2=0xA0;

sbit LATCH1=P2^1;
sbit LATCH2=P2^2;

void DelayUs2x(unsigned char t);
void DelayMs(unsigned char t);

void Display(unsigned long int Number)
{
	unsigned int segment_code[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
	unsigned int bit_code[8]={0x7f,0xbf,0xdf,0xef,0xf7,0xfb,0xfd,0xfe};

	unsigned int x[9];
	unsigned int i,j;
	
	for(i=1;Number>=1;i++)
	{
		x[i]=Number%10;
		Number=(Number-x[i])/10;
	}
	j=i;
	
	for(i=0;i<j;i++)
	{
		if(i==0)
		{
			DataPort=0x7f;
			LATCH2=1;
			LATCH2=0;
		
			DataPort=0x00;
			LATCH1=1;
			LATCH1=0;

			DataPort=0;
			LATCH1=1;
			LATCH1=0;
			
			DelayMs(1);
		}
		else
		{
			DataPort=bit_code[i-1];
			LATCH2=1;
			LATCH2=0;
	
			DataPort=segment_code[x[i]];
			LATCH1=1;
			LATCH1=0;
			
			DelayMs(1);
		}
	}
}

void DelayUs2x(unsigned char t)
{
	while(--t);
}

void DelayMs(unsigned char t)
{
	while(t--)
	{
		DelayUs2x(245);
		DelayUs2x(245);
	}
}

51计算器程序#include #include #define uchar unsigned char #define uint unsigned int sbit lcden=P3^4; sbit lcdrw=P3^6; sbit lcdrs=P3^5; uchar num,temp,jia=0,cheng=0,chu=0,jian=0,qing=0; uint key,key1,shu; uchar fuhao,flag1,flag=0; uchar table[]={ 0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x00,}; void delay(uint z) { uint x,y; for(x=z;x>0;x--) for(y=114;y>0;y--); } bit lcd_bz() { bit result; lcdrs = 0; lcdrw= 1; lcden = 1; _nop_(); _nop_(); _nop_(); _nop_(); result=(bit)(P0&0x80;); lcden=0; return result; } void write_com(uchar com) { while(lcd_bz()); lcdrs=0; lcden=0; lcdrw=0; P0=com; delay(5); lcden=1; delay(5); lcden=0; } void write_date(uchar date) { while(lcd_bz()); lcdrs=1; lcden=0; lcdrw=0; P0=date; delay(5); lcden=1; delay(5); lcden=0; } void lcd_init() { lcden=0; write_com(0x38); write_com(0x0c); write_com(0x06); write_com(0x01); } void keyscan() { P3=0xfe; temp=P3; temp=temp&0xf0; while(temp!=0xf0) { delay(5); temp=P3; temp=temp&0xf0; while(temp!=0xf0) { temp=P3; switch(temp) { case 0xee:key=1;num=0;break; case 0xde:key=2;num=0;break; case 0xbe:key=3;num=0;break; case 0x7e:num=1;break; //¼ÓºÅ } while(temp!=0xf0) { temp=P3; temp=temp&0xf0; } } } P3=0xfd; temp=P3; temp=temp&0xf0; while(temp!=0xf0) { delay(5); temp=P3; temp=temp&0xf0; while(temp!=0xf0) { temp=P3; switch(temp) { case 0xed:key=4;num=0;break; case 0xdd:key=5;num=0;break; case 0xbd:key=6;num=0;break; case 0x7d:num=2;break; //¼õºÅ } while( temp!=0xf0) { temp=P3; temp=temp&0xf0; } } } P3=0xfb; temp=P3; temp=temp&0xf0; while(temp!=0xf0) { delay(5); temp=P3; temp=temp&0xf0; while(temp!=0xf0) { temp=P3; switch(temp) { case 0xeb:key=7;num=0;break; case 0xdb:key=8;num=0;break; case 0xbb:key=9;num=0;break
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

fangz688

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值