找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2842|回复: 3
收起左侧

单片机多进制计算器程序+Proteus仿真

[复制链接]
ID:425208 发表于 2020-12-30 19:13 | 显示全部楼层 |阅读模式
可以实现二进制、八进制、十进制计算,进制之间可以通过按钮点击转换
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
2.png 51hei图片20201230191217.png
单片机源程序如下:
  1. #include <reg51.h>
  2. #include <stdio.h>
  3. #include <intrins.h>
  4. #include <string.h>
  5. #include <math.h>
  6. #include <stdlib.h>


  7. #define u8  unsigned char
  8. #define u16  unsigned char
  9. sbit LCDEN=P3^4;
  10. sbit RS=P3^5;
  11. sbit RW=P3^6;
  12. sbit BF=P0^7;
  13. sbit change_m=P3^7;

  14. u8 code keyval[]="789/456*123-c0=+"; //按键对应的符号
  15. u8 data1[10];
  16. u8 k=0;
  17. char  m[8]={0};
  18. double sum=0;
  19. int flag;
  20. void WrDatLCD(unsigned char DatVal);
  21. void WrComLCD(unsigned char ComVal);
  22. void delay(u16 x)          //延时x毫秒
  23. {
  24. u16 i,j;
  25. for(i=0;i<x;i++)
  26.         for(j=0;j<115;j++)
  27.                 ;
  28. }
  29. int convertBinaryToDecimal(long n)
  30. {
  31.         int decimalNumber = 0, i = 0, remainder;
  32.         while (n != 0)
  33.         {
  34.                 remainder = n % 10;
  35.                 n /= 10;
  36.                 decimalNumber += remainder * pow(2, i);
  37.                 ++i;
  38.         }
  39.         return decimalNumber;
  40. }

  41. int convertBinaryToDecimal8(long n)
  42. {
  43.         int decimalNumber = 0, i = 0, remainder;
  44.         while (n != 0)
  45.         {
  46.                 remainder = n % 10;
  47.                 n /= 10;
  48.                 decimalNumber += remainder * pow(8, i);
  49.                 ++i;
  50.         }
  51.         return decimalNumber;
  52. }





  53. u8 keypad4_4()//按键扫描函数:要去抖,若有按键按下,返回对应的按键值(0-15),没有按键按下返回16
  54. {
  55. u8 i,row,temp;
  56. u8 key=16;//按键号,初值设置为16,目的是:没有按键按下时返回16;
  57.           //若不设初值(默认值为0),没有按键按下时,将返回0,会误认为0被按下  
  58. row=0xef; //从第一列开始      
  59. for(i=0;i<4;i++)
  60. {
  61.         P1=0xff;  
  62.         P1=row;        //第i列信号,对应列为低,其他全为高
  63.         row=_crol_(row,1);           //生成下一列信号
  64.         temp=P1; //读入扫描信号
  65.         temp=temp&0x0f; //屏蔽高4位列信号,只保留低4位行信号
  66.         if(temp!=0x0f)//有按键被按下,因为第i列某行有按键按下,则低4位中有一位为低  
  67.         {  
  68.                 delay(20);  //延时去抖
  69.                 temp=P1;  
  70.                 temp=temp&0x0f;  
  71.                 if(temp!=0x0f)   //再次确认有按键被按下
  72.                   {  
  73.                 switch(temp)  //根据低4位行信号,判断哪个按键被按下
  74.             {  
  75.                     case 0x0e:key=0+i;break; //第i列第1行按键被按下
  76.                 case 0x0d:key=4+i;break; //第i列第2行按键被按下  
  77.                 case 0x0b:key=8+i;break; //第i列第3行按键被按下
  78.                                 case 0x07:key=12+i;      //第i列第4行按键被按下
  79.             }
  80.                        
  81.                         do
  82.                         {
  83.                                 temp=P1;              //再次扫描按键
  84.                                   temp=temp&0x0f;  
  85.                           }while(temp!=0x0f); //等待按键释放   
  86.                   }  
  87.      }
  88.        
  89.        
  90.                  if(change_m == 0)
  91.                  {
  92.                          delay(50);
  93.                          if(change_m == 0)
  94.                          {
  95.                                  flag++;
  96.                                  if(flag == 3)
  97.                                  {
  98.                                          flag = 0;
  99.                                  }
  100.                          } while(!change_m);
  101.                  }



  102.                  
  103. }  
  104. return(key);//扫面结束,返回按键值
  105. }

  106. unsigned char DectectBusyBit(void)//状态判断函数(忙/闲?)
  107. {   
  108.         bit result;
  109.         P0 = 0xff;        //读状态前先置高电平,防止误判
  110.         RS = 0;
  111.         delay(5);
  112.     RW = 1;
  113.         LCDEN = 1;
  114.         delay(5);
  115.         result=BF; //若LCM忙,则反复测试,在此处原地踏步;当LCM闲时,才往下继续
  116.         LCDEN = 0;
  117.         return result;                     
  118. }

  119. void WrComLCD(unsigned char ComVal)//写命令函数
  120. {
  121.         while(DectectBusyBit()==1);         //先检测LCM是否空闲
  122.         RS = 0;
  123.         delay(1);
  124.     RW = 0;
  125.         LCDEN = 1;
  126.         P0 = ComVal;
  127.         delay(1);
  128.         LCDEN = 0;       
  129. }

  130. void WrDatLCD(unsigned char DatVal)//写数据函数
  131. {
  132. while(DectectBusyBit()==1);
  133.         RS = 1;
  134.         delay(1);
  135.     RW = 0;
  136.         LCDEN = 1;
  137.         P0 = DatVal;
  138.         delay(1);
  139.         LCDEN = 0;       
  140. }

  141. void LCD_Init(void)//1602初始化函数
  142. {
  143.         WrComLCD(0x38);     // 功能设定:16*2行、5*7点阵、8位数据接口
  144.         WrComLCD(0x38);
  145.         WrComLCD(0x38);   
  146. //多次重复设定功能指令,因为LCD启动后并不知道使用的是4位数据接口还是8位的,所以开始时总是默认为4位
  147.         WrComLCD(0x01);    // 清屏
  148.         WrComLCD(0x06);    // 光标自增、屏幕不动  
  149.         delay(1);              // 延时,等待上面的指令生效,下面再显示,防止出现乱码
  150.         WrComLCD(0x0c);    // 开显示
  151. }
  152.                          
  153. void compute(){
  154.         u8 i,j=0,k,n=0;
  155.         char data3[3]={0};
  156.         int sum1,data2[4]={0};
  157.         int a,b,c,d,o;
  158.         int getValue[6]={0};
  159.         sum=0;

  160.         for(i=0;data1[i]!='\0';i++){
  161.                   if(data1[i]!='+' && data1[i]!='-' && data1[i]!='*' && data1[i]!='/'){
  162.                           data2[j] =data2[j]*10+(data1[i]-'0');

  163.                   }
  164.                   else{
  165.                           data3[n++] = data1[i];
  166.                           j++;
  167.                   }
  168.         }
  169.         a=data2[0];
  170.         b=data2[1];
  171.         c=data2[2];
  172.         d=data2[3];
  173.         if(flag == 1)  //如果二进制
  174.         {
  175.         a=convertBinaryToDecimal(a);
  176.         b=convertBinaryToDecimal(b);
  177.         c=convertBinaryToDecimal(c);
  178.         d=convertBinaryToDecimal(d);
  179.         }
  180.         if(flag == 2)  //如果8进制
  181.         {
  182.         a=convertBinaryToDecimal8(a);
  183.         b=convertBinaryToDecimal8(b);
  184.         c=convertBinaryToDecimal8(c);
  185.         d=convertBinaryToDecimal8(d);
  186.         }

  187.        
  188.         for(i=0;i<n;i++){
  189.                 if(i==0){
  190.                         if(data3[0]=='+')  sum = a + b;
  191.                         if(data3[0]=='-')  sum = a - b;
  192.                         if(data3[0]=='*')  sum = a * b;
  193.                         if(data3[0]=='/')  sum = a / (double)b;
  194.                 }
  195.                 if(i==1){
  196.                         if(data3[1]=='+')  sum = sum+c;
  197.                         if(data3[1]=='-')  sum = sum-c;
  198.                         if(data3[1]=='*')  sum = sum*c;
  199.                         if(data3[1]=='/')  sum = sum/((float)c);
  200.                 }
  201.                 if(i==2){
  202.                         if(data3[2]=='+')  sum = sum+d;
  203.                         if(data3[2]=='-')  sum = sum-d;
  204.                         if(data3[2]=='*')  sum = sum*d;
  205.                         if(data3[2]=='/')  sum = sum/((float)d);
  206.                 }

  207.        
  208.         }

  209.         //判断是小数输出还是整数输出
  210.         sum1 = sum;
  211.         if(sum1==sum){
  212.                
  213.                 switch(flag)
  214.                 {
  215.                         case 0:sprintf(m,"%d",sum1);break;
  216.                         case 2:sprintf(m,"%o",sum1);break;
  217.                         case 1:
  218.         for(a=0;a<6;a++)
  219.         {
  220.                 getValue[a]=0;
  221.         }
  222.         a=0;
  223.         while(sum1!=0)
  224.         {
  225.                
  226.                 o=sum1/2;
  227.                 a++;
  228.                 getValue[a]=sum1-o*2;
  229.                 sum1=sum1/2;
  230.         }               
  231.                         for(b=a;b>=1;b--)
  232.         {
  233.                         WrDatLCD('0'+getValue[b]);
  234.         }
  235.         break;
  236.                         default:break;
  237.                 }
  238.         }
  239.         else{
  240.                 sprintf(m,"%f",sum);
  241.         }

  242.         if(flag!=1){
  243.         //把结果输出出来
  244.         for(k=0;m[k]!='\0';k++){
  245.          WrDatLCD(m[k]);
  246.         }
  247. }

  248.         WrComLCD(0x80+0x40+15);
  249.         switch(flag)
  250.         {
  251.                 case 0:                WrDatLCD('D');break;
  252.                 case 1:                WrDatLCD('B');break;
  253.                 case 2:                WrDatLCD('O');break;               
  254.         }
  255. }  





  256. void main()
  257. {
  258.         int y;       
  259.         LCD_Init();
  260.         delay(5);   //延时,等待初始化完成
  261.         WrDatLCD('0');
  262.                 WrComLCD(0x80);           //设置显示地址第一行第一位:0X00(0x80+0x00)

  263.         while(1){

  264.        
  265.                 y= keypad4_4();
  266.                 if(y==12)  { k=0;WrComLCD(0x01);WrDatLCD('0'); WrComLCD(0x80);}   //清屏
  267. ……………………

  268. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

所有资料51hei提供下载:
计算器.zip (96.7 KB, 下载次数: 83)

评分

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

查看全部评分

回复

使用道具 举报

ID:971250 发表于 2021-10-11 23:56 来自手机 | 显示全部楼层
请问进制之间怎么转换
回复

使用道具 举报

ID:971940 发表于 2021-10-14 14:09 | 显示全部楼层

请问进制之间怎么转换
回复

使用道具 举报

ID:971940 发表于 2021-10-14 14:49 | 显示全部楼层
实现二进制、八进制、十进制计算,进制之间可以通过按钮点击转换?可以通过哪个按钮呀
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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