基于51单片机的简易计算器设计 单片机型号选择STC89C52RC,读取16*16矩阵式键盘对应的键值,可以进行五位数的加、减、乘、除运算,计算结果通过LCD1602显示,并可通过AT24C02存储模块对计算结果进行存储与调用。
实物演示
单片机源程序如下:
- #include <reg52.h>
- #define uchar unsigned char
- #define uint unsigned int
- sbit rs=P2^5; //指令or数据
- sbit wela=P2^6; //读or写
- sbit lcden=P2^7; //使能信号
- sbit K1 = P2^3;
- sbit K2 = P2^2;
- sbit I2C_SCL = P2^1;
- sbit I2C_SDA = P2^0;
- uchar code table[]= " ";
- //uchar code XUEHAO[]= "190302051";
- long int data_a,data_b; //第一个数和第二个数
- long int data_c; //计算结果
- uchar dispaly[10]; //显示缓冲
- //--声明全局变量--//
- void I2C_Delay10us();
- void I2C_Start(); //起始信号:在I2C_SCL时钟信号在高电平期间I2C_SDA信号产生一个下降沿
- void I2C_Stop(); //终止信号:在I2C_SCL时钟信号高电平期间I2C_SDA信号产生一个上升沿
- uchar I2C_SendByte(uchar dat, uchar ack);//使用I2c读取一个字节
- uchar I2C_ReadByte(); //通过I2C发送一个字节。在I2C_SCL时钟信号高电平期间,保持发送信号I2C_SDA保持稳定
- void I2C_Delay10us()
- {
- uchar a, b;
- for(b=1; b>0; b--)
- {
- for(a=2; a>0; a--);
- }
- }
- void I2C_Start()
- {
- I2C_SDA = 1;
- I2C_Delay10us();
- I2C_SCL = 1;
- I2C_Delay10us();//建立时间是I2C_SDA保持时间>4.7us
- I2C_SDA = 0;
- I2C_Delay10us();//保持时间是>4us
- I2C_SCL = 0;
- I2C_Delay10us();
- }
- void I2C_Stop()
- {
- I2C_SDA = 0;
- I2C_Delay10us();
- I2C_SCL = 1;
- I2C_Delay10us();//建立时间大于4.7us
- I2C_SDA = 1;
- I2C_Delay10us();
- }
- uchar I2C_SendByte(uchar dat, uchar ack)
- {
- uchar a = 0,b = 0;//最大255,一个机器周期为1us,最大延时255us。
- for(a=0; a<8; a++)//要发送8位,从最高位开始
- {
- I2C_SDA = dat >> 7;
- dat = dat << 1;
- I2C_Delay10us();
- I2C_SCL = 1;
- I2C_Delay10us();
- I2C_SCL = 0;
- I2C_Delay10us();
- }
- I2C_SDA = 1;
- I2C_Delay10us();
- I2C_SCL = 1;
- while(I2C_SDA && (ack == 1))
- {
- b++;
- if(b > 200)
- {
- I2C_SCL = 0;
- I2C_Delay10us();
- return 0;
- }
- }
- I2C_SCL = 0;
- I2C_Delay10us();
- return 1;
- }
- uchar I2C_ReadByte()
- {
- uchar a = 0,dat = 0;
- I2C_SDA = 1; //起始和发送一个字节之后I2C_SCL都是0
- I2C_Delay10us();
- for(a=0; a<8; a++)//接收8个字节
- {
- I2C_SCL = 1;
- I2C_Delay10us();
- dat <<= 1;
- dat |= I2C_SDA;
- I2C_Delay10us();
- I2C_SCL = 0;
- I2C_Delay10us();
- }
- return dat;
- }
- void AT24C02Write(unsigned char addr,unsigned char dat)
- {
- I2C_Start();
- I2C_SendByte(0xa0, 1);//发送写器件地址
- I2C_SendByte(addr, 1);//发送要写入内存地址
- I2C_SendByte(dat, 0); //发送数据
- I2C_Stop();
- }
- unsigned char AT24C02Read(unsigned char addr)
- {
- unsigned char num;
- I2C_Start();
- I2C_SendByte(0xa0, 1); //发送写器件地址
- I2C_SendByte(addr, 1); //发送要读取的地址
- I2C_Start();
- I2C_SendByte(0xa1, 1); //发送读器件地址
- num=I2C_ReadByte(); //读取数据
- I2C_Stop();
- return num;
- }
- void LCD_Delay_us(unsigned int t)
- {
- while(t--); //t=0,退出
- }
- void LCD_Delay_ms(unsigned int t)
- {
- unsigned int i,j;
- for(i=0;i<t;i++) //执行t次循环
- for(j=0;j<113;j++) //执行113次循环
- ;
- }
- void write_com(uchar com) //1602液晶写指令
- {
- rs=0; //写指令
- lcden=0; //使能1602
- P0=com; //写入指令com
- LCD_Delay_ms(1); //延时1ms
- lcden=1; //使能1602
- LCD_Delay_ms(2); //延时2ms
- lcden=0; //使能1602
- }
- void write_date(uchar date) //1602液晶写数据
- {
- rs=1; //写数据
- lcden=0; //使能1602
- P0=date; //写入数据date
- LCD_Delay_ms(1); //延时1ms
- lcden=1; //使能1602
- LCD_Delay_ms(2); //延时2ms
- lcden=0; //使能1602
- }
- void W_lcd(unsigned char x,unsigned char y,unsigned char Data)
- {
- if (y == 0){write_com(0x80 + x);} //第一行
- else{write_com(0xc0 + x);} //第二行
- write_date( Data); //写入数据
- }
- //指定x,y写入字符串函数
- void LCD_Write_String(unsigned char x,unsigned char y,unsigned char *s)
- {
- if (y == 0){write_com(0x80 + x);} //第一行
- else{write_com(0xC0 + x);} //第二行
- while (*s) //
- {write_date( *s); s++;} //写入数据
- }
- void init_lcd(void) //初始化液晶,及画面初始化
- {
- wela=0; //写液晶
- lcden=0; //使能1602
- write_com(0x38); //8 位总线,双行显示,5X7 的点阵字符
- LCD_Delay_us(100); //延时100us
- write_com(0x0c); //开显示,无光标,光标不闪烁
- write_com(0x06); //光标右移动
- write_com(0x01); //清屏
- write_com(0x80); //DDRAM 地址归0
- }
- short keycheckdown() /* 反转法键盘扫描 */
- {
- short temp1,temp2,temp,a=0xff;
- P1=0xf0; /* 输入行值(或列值) */
- LCD_Delay_ms(20); /* 延时 */
- temp1=P1; /* 读列值(或行值) */
- P1=0xff;
- LCD_Delay_ms(20); /* 延时 */
- P1=0x0f; /* 输入列值(或行值) */
- LCD_Delay_ms(20); /* 延时 */
- temp2=P1; /* 读行值(或列值) */
- P1=0xff;
- temp=(temp1&0xf0)|(temp2&0xf); /* 将两次读入数据组合 */
- switch(temp) /* 通过读入数据组合判断按键位置 */
- {
- case 0x77 :a=0x0d;break;// 按键/
- case 0x7b :a=0x0e; break;// 按键=
- case 0x7d :a=0; break;// 按键0
- case 0x7e :a=0x0f; break;// 按键CE
- case 0xb7 :a=0x0c;break;// 按键*
- case 0xbb :a=0x9;break; // 按键9
- case 0xbd :a=0x8;break; // 按键8
- case 0xbe :a=0x7;break; // 按键7
- case 0xd7 :a=0x0b;break;// 按键-
- case 0xdb :a=0x6;break; // 按键6
- case 0xdd :a=0x5;break; // 按键5
- case 0xde :a=0x4;break; // 按键4
- case 0xe7 :a=0x0a; break;// 按键+
- case 0xeb :a=3;break; // 按键3
- case 0xed :a=2;break; // 按键2
- case 0xee :a=1;break; // 按键1
- default :a=0xff;
- }
- return a; /* 返回按键值 */
- }
- void display_a() //显示数据a
- {
- dispaly[4]=data_a%100000/10000;
- dispaly[3]=data_a%10000/1000; //千
- dispaly[2]=data_a%1000/100; //百
- dispaly[1]=data_a%100/10; //十
- dispaly[0]=data_a%10; //个
- write_com(0x80+0); //显示数据a
- if(data_a>9999){ write_date('0'+dispaly[4]); }
- if(data_a>999){ write_date('0'+dispaly[3]);} //显示千位
- if(data_a>99){ write_date('0'+dispaly[2]);} //显示百位
- if(data_a>9){ write_date('0'+dispaly[1]);} //显示十位
- write_date('0'+dispaly[0]); //显示个位
- }
- void display_b() //显示数据b
- {
- write_com(0x80+7); //第一行
- dispaly[4]=data_b%100000/10000;
- dispaly[3]=data_b%10000/1000; //千
- dispaly[2]=data_b%1000/100; //百
- dispaly[1]=data_b%100/10; //十
- dispaly[0]=data_b%10; //个
- if(data_b>9999){ write_date('0'+dispaly[4]); }
- if(data_b>999){ write_date('0'+dispaly[3]); } //显示千位
- if(data_b>99) { write_date('0'+dispaly[2]); } //显示百位
- if(data_b>9) { write_date('0'+dispaly[1]); } //显示十位
- write_date('0'+dispaly[0]); //显示个位
- }
- void display_c(x)
- {
- if(data_c<100000000&&data_c>-1)//溢出时显示错误
- {
- dispaly[8]=data_c%1000000000/100000000; //万万
- dispaly[7]=data_c%100000000/10000000; //千万
- dispaly[6]=data_c%10000000/1000000; //百万
- dispaly[5]=data_c%1000000/100000; //十万
- dispaly[4]=data_c%100000/10000; //万
- dispaly[3]=data_c%10000/1000; //千
- dispaly[2]=data_c%1000/100; //百
- dispaly[1]=data_c%100/10; //十
- dispaly[0]=data_c%10; //个
- write_com(0x80+6+0x40); //第一行
- if(x==4)
- {
- if(data_c>99999999) { write_date('0'+dispaly[8]);} //显示万万
- if(data_c>9999999) { write_date('0'+dispaly[7]);} //千万
- if(data_c>999999) { write_date('0'+dispaly[6]);} //百万
- if(data_c>99999) { write_date('0'+dispaly[5]);} //十万
- write_date('0'+dispaly[4]); //万
- write_date('.');
- write_date('0'+dispaly[3]); //千
- write_date('0'+dispaly[2]); //百
- write_date('0'+dispaly[1]); //十
- write_date('0'+dispaly[0]); //个
- }
- else{
- if(data_c>99999999) { write_date('0'+dispaly[8]);} //显示万万
- if(data_c>9999999) { write_date('0'+dispaly[7]);} //千万
- if(data_c>999999) { write_date('0'+dispaly[6]);} //百万
- if(data_c>99999) { write_date('0'+dispaly[5]);} //十万
- if(data_c>9999) { write_date('0'+dispaly[4]);} //万
- if(data_c>999) { write_date('0'+dispaly[3]);} //千
- if(data_c>99) { write_date('0'+dispaly[2]);} //百
- if(data_c>9) { write_date('0'+dispaly[1]);} //十
- write_date('0'+dispaly[0]); //个
- }
- }
- else //溢出时显示错误
- {
- write_com(0x80+11+0x40); //第一行
- write_date('E'); //显示 E
- write_date('r'); //显示R
- write_date('r'); //显示R
- write_date('o'); //显示O
- write_date('r'); //显示E
- }
- }
复制代码
原理图: 无
仿真: 无
代码:
程序代码.7z
(34.48 KB, 下载次数: 11)
|