找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1658|回复: 0
收起左侧

简易单片机计算器程序设计 LCD1602显示

[复制链接]
ID:979771 发表于 2023-11-28 10:27 | 显示全部楼层 |阅读模式
基于51单片机的简易计算器设计  单片机型号选择STC89C52RC,读取16*16矩阵式键盘对应的键值,可以进行五位数的加、减、乘、除运算,计算结果通过LCD1602显示,并可通过AT24C02存储模块对计算结果进行存储与调用。

实物演示

实物演示

单片机源程序如下:
  1. #include <reg52.h>
  2. #define        uchar unsigned char        
  3. #define        uint  unsigned int
  4. sbit rs=P2^5;        //指令or数据
  5. sbit wela=P2^6;      //读or写
  6. sbit lcden=P2^7;         //使能信号
  7. sbit K1 = P2^3;
  8. sbit K2 = P2^2;
  9. sbit I2C_SCL = P2^1;
  10. sbit I2C_SDA = P2^0;
  11. uchar code table[]= "                ";
  12. //uchar code XUEHAO[]= "190302051";
  13. long  int data_a,data_b;         //第一个数和第二个数
  14. long  int data_c;                        //计算结果
  15. uchar dispaly[10];         //显示缓冲
  16. //--声明全局变量--//
  17. void I2C_Delay10us();
  18. void I2C_Start();           //起始信号:在I2C_SCL时钟信号在高电平期间I2C_SDA信号产生一个下降沿
  19. void I2C_Stop();            //终止信号:在I2C_SCL时钟信号高电平期间I2C_SDA信号产生一个上升沿
  20. uchar I2C_SendByte(uchar dat, uchar ack);//使用I2c读取一个字节
  21. uchar I2C_ReadByte();       //通过I2C发送一个字节。在I2C_SCL时钟信号高电平期间,保持发送信号I2C_SDA保持稳定
  22. void I2C_Delay10us()
  23. {
  24.         uchar a, b;
  25.         for(b=1; b>0; b--)
  26.         {
  27.                 for(a=2; a>0; a--);
  28.         }
  29. }
  30. void I2C_Start()
  31. {
  32.         I2C_SDA = 1;
  33.         I2C_Delay10us();
  34.         I2C_SCL = 1;
  35.         I2C_Delay10us();//建立时间是I2C_SDA保持时间>4.7us
  36.         I2C_SDA = 0;
  37.         I2C_Delay10us();//保持时间是>4us
  38.         I2C_SCL = 0;                        
  39.         I2C_Delay10us();               
  40. }
  41. void I2C_Stop()
  42. {
  43.         I2C_SDA = 0;
  44.         I2C_Delay10us();
  45.         I2C_SCL = 1;
  46.         I2C_Delay10us();//建立时间大于4.7us
  47.         I2C_SDA = 1;
  48.         I2C_Delay10us();               
  49. }
  50. uchar I2C_SendByte(uchar dat, uchar ack)
  51. {
  52.         uchar a = 0,b = 0;//最大255,一个机器周期为1us,最大延时255us。                        
  53.         for(a=0; a<8; a++)//要发送8位,从最高位开始
  54.         {
  55.                 I2C_SDA = dat >> 7;         
  56.                 dat = dat << 1;
  57.                 I2C_Delay10us();
  58.                 I2C_SCL = 1;
  59.                 I2C_Delay10us();
  60.                 I2C_SCL = 0;
  61.                 I2C_Delay10us();        
  62.         }
  63.         I2C_SDA = 1;
  64.         I2C_Delay10us();
  65.         I2C_SCL = 1;
  66.         while(I2C_SDA && (ack == 1))
  67.         {
  68.                 b++;
  69.                 if(b > 200)         
  70.                 {
  71.                         I2C_SCL = 0;
  72.                         I2C_Delay10us();
  73.                         return 0;
  74.                 }
  75.         }
  76.         I2C_SCL = 0;
  77.         I2C_Delay10us();
  78.          return 1;               
  79. }

  80. uchar I2C_ReadByte()
  81. {
  82.         uchar a = 0,dat = 0;
  83.         I2C_SDA = 1;                        //起始和发送一个字节之后I2C_SCL都是0
  84.         I2C_Delay10us();
  85.         for(a=0; a<8; a++)//接收8个字节
  86.         {
  87.                 I2C_SCL = 1;
  88.                 I2C_Delay10us();
  89.                 dat <<= 1;
  90.                 dat |= I2C_SDA;
  91.                 I2C_Delay10us();
  92.                 I2C_SCL = 0;
  93.                 I2C_Delay10us();
  94.         }
  95.         return dat;               
  96. }
  97. void AT24C02Write(unsigned char addr,unsigned char dat)
  98. {
  99.         I2C_Start();
  100.         I2C_SendByte(0xa0, 1);//发送写器件地址
  101.         I2C_SendByte(addr, 1);//发送要写入内存地址
  102.         I2C_SendByte(dat, 0);        //发送数据
  103.         I2C_Stop();
  104. }
  105. unsigned char AT24C02Read(unsigned char addr)
  106. {
  107.         unsigned char num;
  108.         I2C_Start();
  109.         I2C_SendByte(0xa0, 1); //发送写器件地址
  110.         I2C_SendByte(addr, 1); //发送要读取的地址
  111.         I2C_Start();
  112.         I2C_SendByte(0xa1, 1); //发送读器件地址
  113.         num=I2C_ReadByte(); //读取数据
  114.         I2C_Stop();
  115.         return num;        
  116. }
  117. void LCD_Delay_us(unsigned int t)
  118. {
  119.         while(t--);           //t=0,退出
  120. }
  121. void LCD_Delay_ms(unsigned int t)
  122. {
  123.         unsigned int i,j;
  124.         for(i=0;i<t;i++)                       //执行t次循环
  125.         for(j=0;j<113;j++)                     //执行113次循环
  126.         ;
  127. }
  128. void write_com(uchar com)            //1602液晶写指令
  129. {
  130.         rs=0;                       //写指令
  131.         lcden=0;                    //使能1602
  132.         P0=com;                     //写入指令com
  133.         LCD_Delay_ms(1);            //延时1ms
  134.         lcden=1;                    //使能1602
  135.         LCD_Delay_ms(2);            //延时2ms  
  136.         lcden=0;                        //使能1602
  137. }
  138. void write_date(uchar date) //1602液晶写数据
  139. {
  140.         rs=1;                        //写数据
  141.         lcden=0;                     //使能1602
  142.         P0=date;                     //写入数据date
  143.         LCD_Delay_ms(1);             //延时1ms
  144.         lcden=1;                     //使能1602
  145.         LCD_Delay_ms(2);             //延时2ms
  146.         lcden=0;                         //使能1602
  147. }
  148. void W_lcd(unsigned char x,unsigned char y,unsigned char Data)
  149. {     
  150.          if (y == 0){write_com(0x80 + x);}   //第一行  
  151.          else{write_com(0xc0 + x);}          //第二行
  152.          write_date( Data);                  //写入数据
  153. }
  154. //指定x,y写入字符串函数
  155. void LCD_Write_String(unsigned char x,unsigned char y,unsigned char *s)
  156. {     
  157. if (y == 0){write_com(0x80 + x);}                 //第一行
  158. else{write_com(0xC0 + x);}                        //第二行
  159. while (*s)                                        //
  160.          {write_date( *s); s++;}                      //写入数据
  161. }

  162. void init_lcd(void)            //初始化液晶,及画面初始化
  163. {
  164.         wela=0;                      //写液晶
  165.         lcden=0;                     //使能1602
  166.         write_com(0x38);             //8 位总线,双行显示,5X7 的点阵字符
  167.         LCD_Delay_us(100);           //延时100us
  168.         write_com(0x0c);             //开显示,无光标,光标不闪烁
  169.         write_com(0x06);             //光标右移动
  170.         write_com(0x01);             //清屏
  171.         write_com(0x80);             //DDRAM 地址归0
  172. }
  173. short keycheckdown()                                /* 反转法键盘扫描 */
  174. {

  175.         short temp1,temp2,temp,a=0xff;
  176.         P1=0xf0;                                                /* 输入行值(或列值) */
  177.         LCD_Delay_ms(20);                                /* 延时 */
  178.         temp1=P1;                                                /* 读列值(或行值) */
  179.         P1=0xff;
  180.         LCD_Delay_ms(20);                                /* 延时 */
  181.         P1=0x0f;                                                /* 输入列值(或行值) */
  182.         LCD_Delay_ms(20);                                /* 延时 */        
  183.         temp2=P1;                                                /* 读行值(或列值) */
  184.         P1=0xff;
  185.         temp=(temp1&0xf0)|(temp2&0xf);        /* 将两次读入数据组合 */
  186.         switch(temp)                                        /* 通过读入数据组合判断按键位置 */
  187.         {


  188.                 case 0x77 :a=0x0d;break;//  按键/   
  189.                 case 0x7b :a=0x0e; break;// 按键=        
  190.                 case 0x7d :a=0;           break;// 按键0
  191.                 case 0x7e :a=0x0f; break;// 按键CE

  192.                 case 0xb7 :a=0x0c;break;//         按键*
  193.                 case 0xbb :a=0x9;break; //  按键9
  194.                 case 0xbd :a=0x8;break; //  按键8
  195.                 case 0xbe :a=0x7;break; //         按键7

  196.                 case 0xd7 :a=0x0b;break;//  按键-
  197.                 case 0xdb :a=0x6;break; //         按键6
  198.                 case 0xdd :a=0x5;break; //         按键5
  199.                 case 0xde :a=0x4;break; //         按键4

  200.                 case 0xe7 :a=0x0a; break;// 按键+
  201.                 case 0xeb :a=3;break;        //  按键3
  202.                 case 0xed :a=2;break;        //  按键2
  203.                 case 0xee :a=1;break;        //  按键1

  204.                 default :a=0xff;
  205.         }
  206.         return a;                                                /* 返回按键值 */
  207. }
  208. void display_a() //显示数据a
  209. {
  210.         dispaly[4]=data_a%100000/10000;
  211.         dispaly[3]=data_a%10000/1000;    //千
  212.         dispaly[2]=data_a%1000/100;      //百
  213.         dispaly[1]=data_a%100/10;        //十
  214.         dispaly[0]=data_a%10;            //个

  215.         write_com(0x80+0);               //显示数据a
  216.         if(data_a>9999){        write_date('0'+dispaly[4]); }
  217.         if(data_a>999){        write_date('0'+dispaly[3]);}      //显示千位
  218.         if(data_a>99){        write_date('0'+dispaly[2]);}                 //显示百位
  219.         if(data_a>9){        write_date('0'+dispaly[1]);}     //显示十位
  220.                               write_date('0'+dispaly[0]);      //显示个位
  221. }

  222. void display_b() //显示数据b
  223. {
  224.         write_com(0x80+7); //第一行
  225.         dispaly[4]=data_b%100000/10000;
  226.         dispaly[3]=data_b%10000/1000;    //千
  227.         dispaly[2]=data_b%1000/100;      //百
  228.         dispaly[1]=data_b%100/10;        //十
  229.         dispaly[0]=data_b%10;            //个
  230. if(data_b>9999){        write_date('0'+dispaly[4]); }
  231. if(data_b>999){        write_date('0'+dispaly[3]); }     //显示千位   
  232. if(data_b>99) {        write_date('0'+dispaly[2]); }    //显示百位  
  233. if(data_b>9)  {        write_date('0'+dispaly[1]); }     //显示十位
  234.                               write_date('0'+dispaly[0]);      //显示个位
  235. }
  236. void display_c(x)
  237. {

  238.         if(data_c<100000000&&data_c>-1)//溢出时显示错误
  239.         {        

  240.                 dispaly[8]=data_c%1000000000/100000000;            //万万
  241.                 dispaly[7]=data_c%100000000/10000000;            //千万
  242.                 dispaly[6]=data_c%10000000/1000000;                    //百万
  243.                 dispaly[5]=data_c%1000000/100000;                    //十万
  244.                 dispaly[4]=data_c%100000/10000;                            //万
  245.                 dispaly[3]=data_c%10000/1000;                            //千
  246.                 dispaly[2]=data_c%1000/100;                              //百
  247.                 dispaly[1]=data_c%100/10;                                //十
  248.                 dispaly[0]=data_c%10;                                    //个
  249.                 write_com(0x80+6+0x40); //第一行
  250.                 if(x==4)
  251.                 {   
  252.                 if(data_c>99999999)        {        write_date('0'+dispaly[8]);}    //显示万万
  253.                 if(data_c>9999999)        {        write_date('0'+dispaly[7]);}   //千万
  254.                 if(data_c>999999)        {        write_date('0'+dispaly[6]);}    //百万
  255.                 if(data_c>99999)        {        write_date('0'+dispaly[5]);}    //十万
  256.                                                                 write_date('0'+dispaly[4]);    //万
  257.                                                                 write_date('.');
  258.                                                                 write_date('0'+dispaly[3]);    //千
  259.                                                                 write_date('0'+dispaly[2]);    //百
  260.                                                                 write_date('0'+dispaly[1]);    //十
  261.                                                                 write_date('0'+dispaly[0]);    //个
  262.                 }
  263.         else{
  264.                 if(data_c>99999999)        {        write_date('0'+dispaly[8]);}    //显示万万
  265.                 if(data_c>9999999)        {        write_date('0'+dispaly[7]);}    //千万
  266.                 if(data_c>999999)                {        write_date('0'+dispaly[6]);}    //百万
  267.                 if(data_c>99999)                {        write_date('0'+dispaly[5]);}    //十万
  268.                 if(data_c>9999)                        {        write_date('0'+dispaly[4]);}    //万
  269.                 if(data_c>999)                        {        write_date('0'+dispaly[3]);}    //千
  270.                 if(data_c>99)                                {        write_date('0'+dispaly[2]);}    //百
  271.                 if(data_c>9)                                {        write_date('0'+dispaly[1]);}    //十
  272.                                                                                                         write_date('0'+dispaly[0]);    //个
  273.                 }
  274.         }
  275.         else  //溢出时显示错误
  276.         {
  277.                 write_com(0x80+11+0x40); //第一行
  278.                 write_date('E');         //显示 E
  279.                 write_date('r');         //显示R
  280.                 write_date('r');                 //显示R
  281.                 write_date('o');                 //显示O
  282.                 write_date('r');                 //显示E
  283.         }
  284. }

复制代码

原理图: 无
仿真: 无
代码: 程序代码.7z (34.48 KB, 下载次数: 11)

评分

参与人数 1黑币 +50 收起 理由
admin + 50

查看全部评分

回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表