找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3983|回复: 9
收起左侧

简易计算器Proteus仿真及单片机源码

[复制链接]
ID:582956 发表于 2019-7-12 14:59 | 显示全部楼层 |阅读模式
上周做的一个简易计算器的仿真,
功能:可以精确小数点后两位,还可以进行十六进制,十进制,八进制,二进制转换

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
0.png 捕获.PNG

单片机源程序如下:
  1. #include<reg51.h>   //头文件
  2. #define uint unsigned int
  3. #define uchar unsigned char

  4. sbit lcden=P1^0; //LCD1602控制引脚
  5. sbit rs=P1^2;
  6. sbit rw=P1^1;

  7. sbit k1=P3^4;   //十六进制转换
  8. sbit k2=P3^5;   //十进制转换
  9. sbit k3=P3^6;   //八进制转换
  10. sbit k4=P3^7;   //二进制转换

  11. char i,j,temp,num,num_1;
  12. long a,b,c;     //a,第一个数 b,第二个数 c,得数
  13. float a_c,b_c;

  14. bit bt1;
  15. uchar flag,fuhao;//flag表示是否有运算符键按下,fuhao表征按下的是哪个运算符
  16. //flag=1表示运算符键按下,flag=0表示运算符键没有按下;
  17. //fuhao=1为加法,fuhao=2为减法,fuhao=3为乘法,fuhao=4为除法。
  18. uchar code q[]={"<<<<huan ying shi yong<<<<"};
  19. uchar code q1[]={"<ji suan qi<"};
  20. uchar code q2[]={"0123456789ABCDEF"};
  21. uchar code table[]={//运算数字输入数组
  22. 7,8,9,0,
  23. 4,5,6,0,
  24. 1,2,3,0,
  25. 0,0,0,0};
  26. uchar code table1[]={        //砗蠼屑淙胂允咀急傅氖?
  27. 7,8,9,0x2f-0x30,        //7,8,9,÷
  28. 4,5,6,0x2a-0x30,        //4, 5, 6,×
  29. 1,2,3,0x2d-0x30,        //1, 2, 3,-
  30. 0x01-0x30,0,0x3d-0x30,0x2b-0x30//C,0,=,+
  31. };
  32. void delay(uchar z) // 延迟函数
  33. {
  34. uchar y;
  35. for(z;z>0;z--)
  36.    for(y=0;y<110;y++);
  37. }
  38. void write_com(uchar com) // 写指令函数
  39. {
  40.         rs=0;
  41.         P0=com;    //com指令付给P0口
  42.         delay(5);lcden=1;delay(5); lcden=0;
  43. }

  44. void write_date(uchar date) // 写数据函数
  45. {
  46.         rs=1; P0=date; delay(5);
  47.         lcden=1; delay(5); lcden=0;
  48. }

  49. void init() //初始化
  50. {
  51.     num=-1;
  52. lcden=1; //使能信号为高电平
  53. rw=0;
  54. write_com(0x38); //8位,2行
  55. delay(5); write_com(0x38); //8位,2行
  56. delay(5); write_com(0x0c); //显示开,光标关,不闪烁*/
  57. delay(1); write_com(0x06); //增量方式不移位 显竟獗暌贫 柚?
  58. delay(1); write_com(0x80); //检测忙信号
  59. delay(1); write_com(0x01); //显示开,光标关,不闪烁
  60. num_1=0;
  61. i=0; j=0;
  62. a=0;     //第一个参与运算的数
  63. b=0;     //第二个参与运算的数
  64. c=0;
  65. flag=0; //flag表示是否有符号键按下,
  66. fuhao=0; // fuhao表征按下的是哪个符号
  67. }
  68. void keyscan() // 键盘扫描程序
  69. {uint ix=0,addr;       

  70.   if(bt1==0)
  71.         {P2=0xfe;
  72.         if(P2!=0xfe)
  73.         {
  74.            delay(20);// 延迟20ms
  75.            if(P2!=0xfe) {  temp=P2&0xf0;
  76.             switch(temp)
  77.             {
  78.              case 0xe0:num=0;   break;        //7
  79.              case 0xd0:num=1;   break;         //8
  80.              case 0xb0:num=2;   break;        //9
  81.              case 0x70:num=3;   break;        //÷
  82.             }
  83.            }   while(P2!=0xfe);
  84.            if(num==0||num==1||num==2)//如果按下的是'7','8'或'9
  85.            {
  86.                if(j!=0){write_com(0x01); j=0;  }
  87.                if(flag==0)//没有按过运算符键
  88.                     { a=a*10+table[num];    } //按下数字存储到a
  89.                     else//如果按过运算符键
  90.                     {    b=b*10+table[num];   }//按下数字存储到b

  91.                  if(a>100|b>100)bt1=1;
  92.            }
  93.            else//如果按下的是'/'            除法
  94.            {
  95.             flag=1;         //按下运算符
  96.             fuhao=4;//4表示除号已按
  97.            }
  98.            i=table1[num];     //数据显示做准备
  99.            write_date(0x30+i);//显示数据或操作符号
  100.         }

  101.         P2=0xfd;
  102.         if(P2!=0xfd)
  103.         {
  104.            delay(20);
  105.            if(P2!=0xfd){  temp=P2&0xf0;
  106.             switch(temp)
  107.             {
  108.              case 0xe0:num=4; break; //4
  109.              case 0xd0:num=5; break; //5
  110.              case 0xb0:num=6; break; //6
  111.              case 0x70:num=7; break; //×
  112.             }
  113.            }   while(P2!=0xfd);//等待按键释放
  114.            if(num==4||num==5||num==6&&num!=7)//如果按下的是'4','5'或'6'
  115.            {
  116.             if(j!=0){ write_com(0x01);  j=0;   }
  117.                if(flag==0)//没有按过运算符键
  118.             { a=a*10+table[num];     }
  119.             else//如果按过运算符键
  120.             { b=b*10+table[num];   }
  121.                    if(a>100|b>100)bt1=1;
  122.            }

  123.            else//如果按下的是'×'
  124.            {  flag=1;
  125.               fuhao=3;//3表示乘号已按
  126.            }
  127.            i=table1[num];      //数据显示做准备
  128.            write_date(0x30+i);//显示数据或操作符号
  129.         }

  130.         P2=0xfb;
  131.         if(P2!=0xfb){  delay(20);
  132.            if(P2!=0xfb) { temp=P2&0xf0;
  133.             switch(temp)
  134.             {
  135.              case 0xe0:num=8;    break;         //1
  136.              case 0xd0:num=9;    break;         //2
  137.              case 0xb0:num=10;   break;         //3
  138.              case 0x70:num=11;   break;         //-
  139.             }
  140.            }   while(P2!=0xfb);
  141.            if(num==8||num==9||num==10)//如果按下的是'1','2'或'3'
  142.            {
  143.             if(j!=0){ write_com(0x01);  j=0;  }
  144.               if(flag==0)//没有按过运算符键
  145.             { a=a*10+table[num];   }
  146.             else//如果按过运算符键
  147.             { b=b*10+table[num]; }
  148.            if(a>100|b>100)bt1=1;
  149.            }
  150.            else if(num==11)//如果按下的是'-'
  151.            {
  152.             flag=1;
  153.             fuhao=2;//2表示减号已按
  154.            }
  155.            i=table1[num];  //数据显示做准备
  156.            write_date(0x30+i);//显示数据或操作符号
  157.         }
  158.          if(a>0&b==0)
  159.          {if(k1==0){write_com(0x80);write_date('0');write_date('X');write_date(q2[a/16]);write_date(q2[a%16]);for(ix=8;ix>0;ix--)write_date(' ');write_com(0x80+4);}
  160.           if(k2==0){write_com(0x80);write_date(q2[a/10]);write_date(q2[a%10]);for(ix=8;ix>0;ix--)write_date(' ');write_com(0x80+2);}
  161.           if(k3==0){write_com(0x80);write_date('0');write_date(q2[a/8]);write_date(q2[a%8]);for(ix=8;ix>0;ix--)write_date(' ');write_com(0x80+3);}
  162.           if(k4==0){write_com(0x80);write_date('0');write_date('B');
  163.           
  164.                 addr = a ;
  165.         for (ix = 0; ix < 8; ix ++)
  166.             {
  167.                 if ((addr<<ix) & 0x80)
  168.                     {
  169.                         write_date('1');
  170.                         }
  171.                 else
  172.                     {
  173.                         write_date('0');
  174.                         }
  175.                 }
  176.           
  177.           }
  178.          }
  179.         }
  180.         else
  181.                                                 {        write_com(0xc0+5);
  182.                                                         write_date('!');write_date('R');write_date('O');
  183.                                                         write_date('R');write_date('R');write_date('E');
  184.                                                 }
  185.         P2=0xf7;
  186.         if(P2!=0xf7){   delay(20);
  187.            if(P2!=0xf7){ temp=P2&0xf0;
  188.             switch(temp)
  189.             {
  190.              case 0xe0:num=13; break;  //数字0   
  191.              case 0xd0:num=12; break;   //清0键  
  192.              case 0xb0:num=14; break;  //等于键   
  193.              case 0x70:num=15; break;  //加
  194.             }                                                                  
  195.                 } while(P2!=0xf7);

  196.            switch(num)
  197.            {
  198.             case 12:{write_com(0x01);a=0;b=0;flag=0;fuhao=0;bt1=0;}//按下的是"清零"
  199.                             break;
  200.             case 13:{                //按下的是"0"
  201.                if(flag==0)//没有按过运算符键
  202.                { a=a*10;  write_date(0x30);     P2=0;       }
  203.                else if(flag>=1)//如果按过运算符键
  204.                {  b=b*10;   write_date(0x30);       }
  205.               }     break;
  206.             case 14:{j=1; //按下等于键,根据运算符号进行不同的算术处理
  207.                    if(fuhao==1)           //加法运算
  208.                            {
  209.                                            write_com(0x80+0x4f);//按下等于键,光标前进至第二行最后一个显示处
  210.                                    write_com(0x04);     //设置从后住前写数据,每写完一个数据,光标后退一格       
  211.                                    c=a+b;
  212.                                    while(c!=0){write_date(0x30+c%10);        c=c/10;                }
  213.                                    write_date(0x3d);     //再写"="
  214.                                    a=0;b=0;flag=0;fuhao=0;
  215.                    }
  216.                       else if(fuhao==2)         //减法运算
  217.                              {
  218.                                         write_com(0x80+0x4f);//光标前进至第二行最后一个显示处
  219.                             write_com(0x04);     //设置从后住前写数据,每写完一个数据,光标后退一格(这个照理说顺序不对,可显示和上段一样)
  220.                                    if(a-b>0)                 c=a-b;
  221.                                    else                      c=b-a;
  222.                                    while(c!=0)        { write_date(0x30+c%10);c=c/10;         }
  223.                                    if(a-b<0)          write_date(0x2d);
  224.                                    write_date(0x3d);     //再写"="           
  225.                                    a=0;b=0;flag=0;fuhao=0;
  226.                    }
  227.                        else if(fuhao==3)        //乘法运算
  228.                                  {write_com(0x80+0x4f);            write_com(0x04);
  229.                              c=a*b;
  230.                              while(c!=0)        {write_date(0x30+c%10);        c=c/10;              }
  231.                              write_date(0x3d);           a=0;b=0;flag=0;fuhao=0;
  232.                      }
  233.                         else if(fuhao==4)//除法运算       
  234.                                           {write_com(0x80+0x4f);
  235.                                     write_com(0x04);
  236.                                     i=0;
  237.                                             c=(long)(((float)a/b)*1000);
  238.                                             while(c!=0)
  239.                                              {
  240.                                                        write_date(0x30+c%10);
  241.                                                         c=c/10;
  242.                                                            i++; if(i==3) write_date(0x2e);
  243.                                                 }
  244.                                                     if(a/b<=0)
  245.                                                          {
  246.                                                                  if(i<=2)
  247.                                                                  {       
  248.                                                                         
  249.                                                                         if(i==1) write_date(0x30);
  250.                                                                         write_date(0x2e);           //  .
  251.                                                                         write_date(0x30);
  252.                                                                  }
  253.                                                                 
  254.                                                       write_date(0x30);           //
  255.                                                         }
  256.                                             write_date(0x3d);                              
  257.                                                     a=0;b=0;flag=0;fuhao=0;
  258.                                        
  259.                          }
  260.               }  break;
  261.              case 15:{write_date(0x30+table1[num]);flag=1;fuhao=1;}     break;        //加键         设置加标志fuhao=1;
  262.            }
  263.         }//P2!=0xf7
  264. }
  265. /*------------------------------------------------
  266.               写入字符串函数
  267. ------------------------------------------------*/
  268. void LCD_Write_String(unsigned char x,unsigned char y,unsigned char *s)
  269. {     
  270. if (y == 0)
  271.         {     
  272.          write_com(0x80 + x);     //表示第一行
  273.         }
  274. else
  275.         {      
  276.         write_com(0xC0 + x);      //表示第二行
  277.         }        
  278. while (*s)
  279.         {     
  280. write_date( *s);     
  281. s ++;     
  282.         }
  283. }

  284. void main()
  285. {uint c=60000;
  286.         init();                //系统初始化
  287.         LCD_Write_String(0,0,q);
  288.         LCD_Write_String(0,1,q1);
  289.         while(c--);
  290.                 init();                //系统初始化
  291.         while(1)
  292.         {                
  293.            keyscan();//键扫描
  294.         }
  295. }
复制代码

所有资料51hei提供下载:
简易计算器.rar (71.24 KB, 下载次数: 74)
回复

使用道具 举报

ID:262 发表于 2022-2-27 21:09 | 显示全部楼层
喋喋phone 发表于 2022-2-27 21:02
我仿真怎么没反应啊?

用Proteus8.8打开就可以了,我帮你转成了8.8的格式: 简易计算器.7z (61.18 KB, 下载次数: 16)
51hei.gif
回复

使用道具 举报

ID:1007108 发表于 2022-2-27 21:02 来自手机 | 显示全部楼层
我仿真怎么没反应啊?
回复

使用道具 举报

ID:1007108 发表于 2022-2-27 21:24 来自手机 | 显示全部楼层
heicad 发表于 2022-2-27 21:09
用Proteus8.8打开就可以了,我帮你转成了8.8的格式

我用8.6版本的可以吗?
回复

使用道具 举报

ID:1007266 发表于 2022-2-28 15:14 | 显示全部楼层
把代码复制粘贴完运行出错是怎么回事
回复

使用道具 举报

ID:1007108 发表于 2022-3-1 21:45 来自手机 | 显示全部楼层
heicad 发表于 2022-2-27 21:09
用Proteus8.8打开就可以了,我帮你转成了8.8的格式:

hex文件可以直接用吗,为什么我按了按键没反应啊?而且显示器什么也没有
回复

使用道具 举报

ID:1007108 发表于 2022-3-1 23:56 | 显示全部楼层
图3.png

我的仿真怎么跟你的不太一样啊
回复

使用道具 举报

ID:262 发表于 2022-3-2 01:03 | 显示全部楼层
你去下载我帮你修改后的Proteus文件 只能用用Proteus8.8这个版本打开,因为其他版本我也没有 无法测试,然后就是把hex文件要选对
回复

使用道具 举报

ID:1007108 发表于 2022-3-3 19:43 来自手机 | 显示全部楼层
heicad 发表于 2022-3-2 01:03
你去下载我帮你修改后的Proteus文件 只能用用Proteus8.8这个版本打开,因为其他版本我也没有 无法测试,然后 ...

大佬,好像还是一样的,lcd屏不显示东西,按键按下去没有效果
回复

使用道具 举报

ID:1007108 发表于 2022-3-3 19:46 来自手机 | 显示全部楼层
heicad 发表于 2022-3-2 01:03
你去下载我帮你修改后的Proteus文件 只能用用Proteus8.8这个版本打开,因为其他版本我也没有 无法测试,然后 ...

有用了,谢谢
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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