找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3447|回复: 3
打印 上一主题 下一主题
收起左侧

单片机计算器仿真程序(注释很全)实现9999以内简单的四则运算

[复制链接]
跳转到指定楼层
楼主
矩阵键盘输入,LCD屏幕显示,可以实现9999以内简单的四则运算

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)


单片机源程序如下,代码有详细的中文注释:
  1. /*
  2. 接盘按键说明:
  3. --------------------------------------------------
  4.             |  7  |  8  |  9  |  /  |  
  5.             - - - - - - - - - - - - -
  6.             |  4  |  5  |  6  |  *  |
  7.             - - - - - - - - - - - - -
  8.             |  1  |  2  |  3  |  -  |
  9.             - - - - - - - - - - - - -
  10.                         | C   |  0  |  =  |  +  |
  11. --------------------------------------------------*/
  12. //操作简介
  13. // 按第一个数,再按'+-*/',再按'='显示出结果,然后按C清屏
  14. // 加最大9999+9999=19998
  15. // 减最大9999-0   =9999
  16. // 乘最大9999*9999=99980001
  17. // 除 1/9=0.1111 保留小数点后4位
  18. #include<reg51.h>
  19. #define uint unsigned int
  20. #define uchar unsigned char
  21. //--------LCD1602-------------------
  22. //P00-07==== D0-7
  23. sbit rs=P2^7;        //指令or数据
  24. sbit wela=P2^6;      //读or写
  25. sbit lcden=P2^5;         //使能信号
  26. //--------LCD1602-------------------
  27. //--------KEY-----------------------
  28. //P1口
  29. //--------KEY-----------------------
  30. uchar code table[]= "                ";

  31. long  int data_a,data_b;         //第一个数和第二个数
  32. long  int data_c;                        //计算结果

  33. uchar dispaly[10];         //显示缓冲

  34. //************************************************************************/
  35. // 描述: 延时t us函数
  36. //************************************************************************/
  37. void LCD_Delay_us(unsigned int t)
  38. {
  39.         while(t--);           //t=0,退出
  40. }
  41. //************************************************************************/
  42. // 描述: 延时t ms函数
  43. //************************************************************************/
  44. void LCD_Delay_ms(unsigned int t)
  45. {
  46.         unsigned int i,j;
  47.         for(i=0;i<t;i++)                       //执行t次循环
  48.         for(j=0;j<113;j++)                     //执行113次循环
  49.         ;
  50. }
  51. //************************************************************************/
  52. // 描述: 1602液晶写指令
  53. //************************************************************************/
  54. void write_com(uchar com)            //1602液晶写指令
  55. {
  56.         rs=0;                       //写指令
  57.         lcden=0;                    //使能1602
  58.         P0=com;                     //写入指令com
  59.         LCD_Delay_ms(1);            //延时1ms
  60.         lcden=1;                    //使能1602
  61.         LCD_Delay_ms(2);            //延时2ms  
  62.         lcden=0;                        //使能1602
  63. }
  64. //************************************************************************/
  65. // 描述:1602液晶写数据
  66. //************************************************************************/
  67. void write_date(uchar date) //1602液晶写数据
  68. {
  69.         rs=1;                        //写数据
  70.         lcden=0;                     //使能1602
  71.         P0=date;                     //写入数据date
  72.         LCD_Delay_ms(1);             //延时1ms
  73.         lcden=1;                     //使能1602
  74.         LCD_Delay_ms(2);             //延时2ms
  75.         lcden=0;                         //使能1602
  76. }
  77. //************************************************************************/
  78. // 描述:指定x,y写入字符函数
  79. //************************************************************************/
  80. void W_lcd(unsigned char x,unsigned char y,unsigned char Data)
  81. {     
  82.          if (y == 0){write_com(0x80 + x);}   //第一行  
  83.          else{write_com(0xc0 + x);}          //第二行
  84.          write_date( Data);                  //写入数据
  85. }
  86. //指定x,y写入字符串函数
  87. void LCD_Write_String(unsigned char x,unsigned char y,unsigned char *s)
  88. {     
  89. if (y == 0){write_com(0x80 + x);}                 //第一行
  90. else{write_com(0xC0 + x);}                        //第二行
  91. while (*s)                                        //
  92.          {write_date( *s); s++;}                      //写入数据
  93. }
  94. //************************************************************************/
  95. // 描述:初始化液晶,及画面初始化
  96. //************************************************************************/
  97. void init_lcd(void)            //初始化液晶,及画面初始化
  98. {
  99.         wela=0;                      //写液晶
  100.         lcden=0;                     //使能1602
  101.         write_com(0x38);             //8 位总线,双行显示,5X7 的点阵字符
  102.         LCD_Delay_us(100);           //延时100us
  103.         write_com(0x0c);             //开显示,无光标,光标不闪烁
  104.         write_com(0x06);             //光标右移动
  105.         write_com(0x01);             //清屏
  106.         write_com(0x80);             //DDRAM 地址归0
  107. }
  108. //************************************************************************/
  109. // 描述: 反转法键盘扫描
  110. //************************************************************************/
  111. short keycheckdown()                                /* 反转法键盘扫描 */
  112. {

  113.         short temp1,temp2,temp,a=0xff;
  114.         P1=0xf0;                                                /* 输入行值(或列值) */
  115.         LCD_Delay_ms(20);                                /* 延时 */
  116.         temp1=P1;                                                /* 读列值(或行值) */
  117.         P1=0xff;
  118.         LCD_Delay_ms(20);                                /* 延时 */
  119.         P1=0x0f;                                                /* 输入列值(或行值) */
  120.         LCD_Delay_ms(20);                                /* 延时 */        
  121.         temp2=P1;                                                /* 读行值(或列值) */
  122.         P1=0xff;
  123.         temp=(temp1&0xf0)|(temp2&0xf);        /* 将两次读入数据组合 */
  124.         switch(temp)                                        /* 通过读入数据组合判断按键位置 */
  125.         {


  126.                 case 0x77 :a=0x0a;break;//  按键+   
  127.                 case 0x7b :a=0x0e; break;// 按键=        
  128.                 case 0x7d :a=0;           break;// 按键0
  129.                 case 0x7e :a=0x0f; break;// 按键CE

  130.                 case 0xe7 :a=0x0d;break;//         按键/
  131.                 case 0xeb :a=0x9;break; //  按键9
  132.                 case 0xed :a=0x8;break; //  按键8
  133.                 case 0xee :a=0x7;break; //         按键7

  134.                 case 0xd7 :a=0x0c;break;//  按键*
  135.                 case 0xdb :a=0x6;break; //         按键6
  136.                 case 0xdd :a=0x5;break; //         按键5
  137.                 case 0xde :a=0x4;break; //         按键4

  138.                 case 0xb7 :a=0x0b; break;// 按键-
  139.                 case 0xbb :a=3;break;        //  按键3
  140.                 case 0xbd :a=2;break;        //  按键2
  141.                 case 0xbe :a=1;break;        //  按键1

  142.                 default :a=0xff;
  143.         }
  144.         return a;                                                /* 返回按键值 */
  145. }
  146. void display_a() //显示数据a
  147. {
  148.         dispaly[3]=data_a%10000/1000;    //千
  149.         dispaly[2]=data_a%1000/100;      //百
  150.         dispaly[1]=data_a%100/10;        //十
  151.         dispaly[0]=data_a%10;            //个

  152.         write_com(0x80+0);               //显示数据a
  153.         if(data_a>999){        write_date('0'+dispaly[3]);}      //显示千位
  154.         if(data_a>99){        write_date('0'+dispaly[2]);}                 //显示百位
  155.         if(data_a>9){        write_date('0'+dispaly[1]);}     //显示十位
  156.                               write_date('0'+dispaly[0]);      //显示个位
  157. }

  158. void display_b() //显示数据b
  159. {
  160.         write_com(0x80+7); //第一行
  161.         dispaly[3]=data_b%10000/1000;    //千
  162.         dispaly[2]=data_b%1000/100;      //百
  163.         dispaly[1]=data_b%100/10;        //十
  164.         dispaly[0]=data_b%10;            //个

  165. if(data_b>999){        write_date('0'+dispaly[3]); }     //显示千位   
  166. if(data_b>99) {        write_date('0'+dispaly[2]); }    //显示百位  
  167. if(data_b>9)  {        write_date('0'+dispaly[1]); }     //显示十位
  168.                               write_date('0'+dispaly[0]);      //显示个位
  169. }

  170. //计算结果
  171. void display_c(x)
  172. {

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

  175.                 dispaly[8]=data_c%1000000000/100000000;            //万万
  176.                 dispaly[7]=data_c%100000000/10000000;            //千万
  177.                 dispaly[6]=data_c%10000000/1000000;                    //百万
  178.                 dispaly[5]=data_c%1000000/100000;                    //十万
  179.                 dispaly[4]=data_c%100000/10000;                            //万
  180.                 dispaly[3]=data_c%10000/1000;                            //千
  181.                 dispaly[2]=data_c%1000/100;                              //百
  182.                 dispaly[1]=data_c%100/10;                                //十
  183.                 dispaly[0]=data_c%10;                                    //个
  184.                 write_com(0x80+6+0x40); //第一行
  185.                 if(x==4)
  186.                 {   
  187.                         if(data_c>99999999)        {        write_date('0'+dispaly[8]);}    //显示万万
  188.                         if(data_c>9999999)        {        write_date('0'+dispaly[7]);}   //千万
  189.                         if(data_c>999999)        {        write_date('0'+dispaly[6]);}    //百万
  190.                         if(data_c>99999)        {        write_date('0'+dispaly[5]);}    //十万
  191.                         if(data_b!=0)                {
  192.                                                                         write_date('0'+dispaly[4]);    //万
  193.                                                                         write_date('.');
  194.                                                                         write_date('0'+dispaly[3]);    //千
  195.                                                                         write_date('0'+dispaly[2]);    //百
  196.                                                                         write_date('0'+dispaly[1]);    //十
  197.                                                                         write_date('0'+dispaly[0]);    //个
  198.                                                                 }
  199.                 }
  200.         else{
  201.                 if(data_c>99999999)        {        write_date('0'+dispaly[8]);}    //显示万万
  202.                 if(data_c>9999999)        {        write_date('0'+dispaly[7]);}    //千万
  203.                 if(data_c>999999)        {        write_date('0'+dispaly[6]);}    //百万
  204.                 if(data_c>99999)        {        write_date('0'+dispaly[5]);}    //十万
  205.                 if(data_c>9999)                {        write_date('0'+dispaly[4]);}    //万
  206.                 if(data_c>999)                {        write_date('0'+dispaly[3]);}    //千
  207.                 if(data_c>99)                {        write_date('0'+dispaly[2]);}    //百
  208.                 if(data_c>9)                {        write_date('0'+dispaly[1]);}    //十
  209.                                                                 write_date('0'+dispaly[0]);    //个
  210.                 }
  211.         }
  212. //        else  //溢出时显示错误
  213. //        {
  214. //                write_com(0x80+11+0x40); //第一行
  215. //                write_date('E');         //显示 E
  216. //                write_date('r');         //显示R
  217. //                write_date('r');                 //显示R
  218. //                write_date('o');                 //显示O
  219. //                write_date('r');                 //显示E
  220. //        }
  221. }

  222. void eql(uchar x)//加减乘除运算
  223. {
  224.         switch(x)                /*功能键选择*/
  225.                         {   
  226.                                 case 1:data_c=data_a+data_b;break;                   //加  /* + S=1 */ /* 数值转换函数 */                                                                        
  227.                                 case 2:if(data_a>=data_b){data_c=data_a-data_b;} /* - S=2 *///减
  228.                        else{data_c=data_b-data_a;W_lcd(5,1,'-');} //负数符号
  229.                                            break;           
  230.                                 case 3:data_c=(data_a*data_b);break;                  /* * S=3 *///乘
  231.                                 case 4:if(data_b==0){LCD_Write_String(0,1,"Error !         ");}else{data_c=(data_a*10000)/data_b;}break;                  /* / S=4 *///除//溢出时显示错误
  232.                                 case 0:break;
  233.                         }         

  234. }

  235. void main()
  236. {
  237.         uchar   key=0xff;               //键值初始化
  238.         uchar   n=0;                                         //第1个数可以按1-4次
  239.         uchar   m=5;                                         //第2个数可以按1-4次
  240.         uchar   x=0;
  241.         data_a=0;                       //前一个数
  242.         data_b=0;                       //后一个数
  243.         data_c=0;                       //结果
  244.         init_lcd();                                          //1602液晶初始化
  245.         display_a();
  246.     while(1)
  247.           {
  248.                   key=keycheckdown();                        /*动态扫描键盘,返回按键对应值,赋给j        key=0到f */
  249.                 if(0xff!=key)                                /*若返回值有效,进入内部处理程序*/
  250.                 {
  251.                   if(key<10)
  252.                   {
  253.                           if(key==0&&n==0) ;
  254.                         else
  255.                         {
  256.                                 if(n<4){data_a=data_a*10+key;m=5;display_a();}n++;  //首先输入第一个数        
  257.                         }
  258.                 if(key==0&&m==0) ;
  259.                         else
  260.                         {
  261.                                 if(m<4){data_b=data_b*10+key;n=5;display_b();}m++;  //必须按了+-*/才能输入第二个数        
  262.                         }
  263.                   }
  264.           else
  265.           {        switch(key)                /*功能键选择*/
  266.                                         {
  267.                                                 case 0xa:n=5;m=0;x=1;W_lcd(5,0,'+');break; //加  /* + S=1 */ /* 数值转换函数 */                                                                        
  268.                                                 case 0xb:n=5;m=0;x=2;W_lcd(5,0,'-');break;                  /* - S=2 *///减
  269.                                                 case 0xc:n=5;m=0;x=3;W_lcd(5,0,'*');break;                  /* * S=3 *///乘
  270.                                                 case 0xd:n=5;m=0;x=4;W_lcd(5,0,'/');break;                  /* / S=4 *///除
  271.                                                 case 0xe:n=5;m=5;eql(x);W_lcd(12,0,'=');display_c(x);break;           /* = */
  272.                                                 case 0xf:n=0;x=0;m=5; data_a=0;data_b=0;data_c=0;LCD_Write_String(0,0,table);LCD_Write_String(0,1,table);W_lcd(0,0,'0');break; /*     C*/
  273.                                         }                           
  274.                   }
  275.                         do{P1=0xf0;}while(P1!=0xf0);                /*等待按键松开*/
  276.                   }//(0xff!=key)         
  277.           }//while
  278. }//main
复制代码

所有资料51hei提供下载:
新建文件夹 (2).zip (83.8 KB, 下载次数: 87)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏2 分享淘帖 顶1 踩
回复

使用道具 举报

沙发
ID:473034 发表于 2019-1-26 10:36 | 只看该作者
一直不太清楚发一个帖子长多少
回复

使用道具 举报

板凳
ID:88256 发表于 2019-9-29 23:00 | 只看该作者
这个给新手学习不错
回复

使用道具 举报

地板
ID:253767 发表于 2019-9-30 08:05 | 只看该作者
谢谢分享!!!
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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