找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 4721|回复: 2
收起左侧

51单片机计算器Proteus仿真+C语言代码

[复制链接]
ID:567237 发表于 2019-6-19 20:39 | 显示全部楼层 |阅读模式
如压缩包
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
000.png

单片机源程序如下:
  1. #include <reg51.h>
  2. #include <stdio.h>
  3. #include <stdlib.h>
  4. #include <string.h>
  5. #include <math.h>
  6. #define uint unsigned int
  7. #define uchar unsigned char
  8. sbit pinRS=P2^0;
  9. sbit pinRW=P2^1;
  10. sbit pinE=P2^2;
  11. #define pindata P0
  12. #define clearscreen() writeinstruc(0x01)
  13. #define cursorreturn() writeinstruc(0x02)
  14. #define inputmode(temp) writeinstruc(temp)
  15. //temp取值如下:
  16. //0x04:减量方式,不移位
  17. //0x05:减量方式,移位
  18. //0x06:增量方式,不移位
  19. //0x07:增量方式,移位
  20. #define dispcontrol(temp) writeinstruc(temp)
  21. //temp取值如下:
  22. //0x08:显示关,光标关,闪烁关
  23. //0x0c:显示关,光标关,闪烁关
  24. //0x0d:显示开,光标关,闪烁开
  25. //0x0e:显示开,光标开,闪烁关
  26. //0x0f:显示开,光标开,闪烁开
  27. #define dispshift(temp) writeinstruc(temp)
  28. //temp取值如下:
  29. //0x10:光标左移
  30. //0x14:光标右移
  31. //0x18:显示整体左移
  32. //0x1c:显示整体右移
  33. #define functionset(temp) writeinstruc(temp)
  34. //temp取值如下:
  35. //0x20:4位,1行,5*7
  36. //0x24:4位,1行,5*10
  37. //0x28:4位,2行,5*7
  38. //0x2c:4位,2行,5*10
  39. //0x30:8位,1行,5*7
  40. //0x34:8位,1行,5*10
  41. //0x38:8位,2行,5*7
  42. //0x3c:8位,2行,5*10
  43. #define setCGRAM_Add(address) writeinstruc(0x40|address)
  44. #define setDDRAM_Add(address) writeinstruc(0x80|address)//短延时
  45. //void shortdelay(uchar i);
  46. //长延时
  47. //void longdelay(uint i);
  48. //P2口初始化
  49. //void initP2(bit i);
  50. //寄存器选择信号
  51. //void setRS(bit i);
  52. //读写操作控制
  53. //void setRW(bit i);
  54. //使能信号
  55. //void setE(bit i);
  56. //读BF以及AC的值
  57. //uint read_BF_AC();
  58. //判忙
  59. //bit statuscheck();
  60. //写指令函数
  61. //void writeinstruc(uint instruc);
  62. //写数据到RAM
  63. //void writedata(uint data1);
  64. //从RAM中读数据函数
  65. //uint readdata(void);
  66. //显示程序 X为行号,Y为列号,date为要显示的数据
  67. //void dispcharacter(int x,uint y,uint data1);
  68. //LCD复位函数
  69. //void LCDreset();
  70. #define uchar unsigned char
  71. uchar k=0,result1;
  72. void shortdelay(uchar i)//短延时
  73. {
  74. for(;i>0;i--);
  75. }
  76. //长延时
  77. void longdelay(uint i)
  78. {
  79. uint j;
  80. for(;i>0;i--)
  81. {for(j=100;j>0;j--);}
  82. }
  83. //延时程序
  84. void delay(int i)
  85. {int j;
  86. for(;i>0;i--)
  87.    for(j=0;j<100;j++);
  88. }
  89. //初试状态
  90. void io_init()
  91. {P1=0xff;
  92. }
  93. //?---------------------------键盘扫描部分-----------------------------//
  94. unsigned char key_scan()
  95. { unsigned char key;
  96.   unsigned char temp;
  97.   uint flag;
  98.   io_init();
  99.    P1=0xf0;
  100.    flag=0;
  101. loop: temp=P1^0xf0;
  102.    while(temp)
  103.     {
  104.      P1=0xfe;
  105.      delay(1);
  106.      if((P1&0xfe)!=0xfe)
  107.      switch(P1)
  108.      {
  109.      case 0xee: key=0x11;flag=1;break;//读7键和值
  110.      case 0xde: key=0x12;flag=1;break;
  111.      case 0xbe: key=0x13;flag=1;break;
  112.      case 0x7e: key=0x14;flag=1;break;//读除键的值
  113.      }
  114.   if(flag)
  115.   goto exit;
  116.   else
  117.      P1=0xfd;
  118.      delay(1);
  119.      if((P1&0xfd)!=0xfd)
  120.      switch(P1)
  121.      {
  122.      case 0xed: key=0x21;flag=1;break;
  123.      case 0xdd: key=0x22;flag=1;break;
  124.      case 0xbd: key=0x23;flag=1;break;
  125.      case 0x7d: key=0x24;flag=1;break; //读4到乘的键值
  126.      }
  127.      if(flag)
  128.   goto exit;
  129.   else
  130.      P1=0xfb;
  131.      delay(1);
  132.      if((P1&0xfb)!=0xfb)
  133.      switch(P1)
  134.      {
  135.      case 0xeb: key=0x31;flag=1;break;
  136.      case 0xdb: key=0x32;flag=1;break;
  137.      case 0xbb: key=0x33;flag=1;break;
  138.      case 0x7b: key=0x34;flag=1;break;//读1到减的键值
  139.      }
  140.      if(flag)
  141.   goto exit;
  142.   else
  143.   P1=0xf7;
  144.      delay(1);
  145.      if((P1&0xf7)!=0xf7)
  146.      switch(P1)
  147.      {
  148.      case 0xe7: key=0x41;flag=1;break;//后退
  149.      case 0xd7: key=0x42;flag=1;break;//0
  150.      case 0xb7: key=0x43;flag=1;break;//等于
  151.      case 0x77: key=0x44;flag=1;break;//加
  152.      }
  153.      
  154. exit: return key;
  155.    }
  156.   if(temp==0)
  157.   goto loop;
  158.   }
  159. //---------------------------------键盘扫描完--------------------------------------------//
  160. //---------------------------------显示初始化?----------------------------------------//
  161. //P0口初始化
  162. void initpindata(bit i)
  163. {
  164. if(i==1) pindata=0xff;
  165. else pindata=0X00;
  166. }
  167. //寄存器选择信号
  168. void setRS(bit i)
  169. {
  170. if(i==1) pinRS=1;
  171. else pinRS=0;
  172. }
  173. //读写操作控制
  174. void setRW(bit i)
  175. {
  176. if(i==1) pinRW=1;
  177. else pinRW=0;
  178. }
  179. //使能信号
  180. void setE(bit i)
  181. {
  182. if(i==1) pinE=1;
  183. else pinE=0;
  184. }
  185. //读BF以及AC的值
  186. uint read_BF_AC()
  187. {
  188. uint temp;
  189. initpindata(1);
  190. setRS(0);
  191. setRW(1);
  192. setE(1);
  193. shortdelay(1);
  194. temp=pindata;
  195. shortdelay(10);
  196. setE(0);
  197. return(temp);
  198. }
  199. //判忙
  200. bit statuscheck()
  201. {
  202. return((bit)(read_BF_AC()&0x80));
  203. }
  204. //写指令函数
  205. void writeinstruc(uint instruc)
  206. {
  207. while(statuscheck());
  208. initpindata(0);
  209. setRS(0);
  210. setRW(0);
  211. setE(0);
  212. pindata=instruc;
  213. //short delay(1);
  214. setE(1);
  215. shortdelay(10);
  216. setE(0);
  217. }
  218. //写数据到RAM
  219. void writedata(uint data1)
  220. {
  221. initpindata(0);
  222. setRS(1);
  223. setRW(0);
  224. setE(0);
  225. pindata=data1;
  226. setE(1);
  227. shortdelay(10);
  228. setE(0);
  229. }
  230. //从RAM中读数据函数
  231. uint readdata(void)
  232. {
  233. uint temp;
  234. initpindata(1);
  235. setRS(1);
  236. setRW(1);
  237. setE(1);
  238. shortdelay(1);
  239. temp=pindata;
  240. shortdelay(10);
  241. setE(0);
  242. return(temp);
  243. }
  244. //显示程序 X为行号,Y为列号,date为要显示的数据
  245. void dispcharacter(int x,uint y,uint data1)
  246. {
  247. uint temp;
  248. while(statuscheck());
  249. temp=y&0x0f;
  250. x&=0x01;
  251. if(x) temp|=0x40;
  252. setDDRAM_Add(temp);
  253. writedata(data1);
  254. // writedata(data1);
  255. }
  256. //LCD复位函数
  257. void LCDreset()
  258. {
  259. clearscreen();
  260. cursorreturn();
  261. }
  262. //LCD初始化
  263. void initLCD()
  264. {
  265. LCDreset();
  266. inputmode(0x06);//增量方式,不移位
  267. dispcontrol(0x0c);//显示开,光标关,闪烁关
  268. functionset(0x38);//8位,2行,5*7
  269. }
  270. //-----------------------------------准备工作完成-----------------------------------//
  271. //-----------------------------------计算器-----------------------------------------//
  272.   compute(char key1,i)
  273. {
  274. signed char m,n,c,act;
  275. long int num1,num2,result;
  276. signed char str[4][4]={{'7','8','9','/'},
  277.                 {'4','5','6','*'},
  278.                 {'1','2','3','-'},
  279.                 {'c','0','=','+'}};
  280. signed char str1[11],string1[2];
  281. m=(key1&0xf0);
  282. m=m/16;
  283. m=m-1;
  284. n=key1&0x0f;
  285. n=n-1;
  286. c=str[m][n];
  287. sprintf(string1,"%c",c);
  288. strcat(str1,string1);
  289. if((c=='+')||(c=='-')||(c=='*')||(c=='/'))
  290. {
  291. act=c;
  292. num1=atoi(str1);
  293. memset(str1,0,11);
  294. memset(string1,0,2);
  295. }
  296. if(c=='=')
  297. {
  298. num2=atoi(str1);

  299. switch(act)
  300. {
  301. case'+':
  302. { if(k==0)
  303.     {result=num1+num2;k++;break;}
  304.    else
  305.    {result+=num2;k++;break;}
  306.    }
  307. case'-':
  308.   {if(k==0)
  309.    {result=num1-num2;k++;break;}
  310.    else
  311.    {result-=num2;k++;break;}
  312.    }
  313. case'*':
  314. {if(k==0)
  315.    {result=num1*num2;k++;break;}
  316.    else
  317.    {result*=num2;k++;break;}
  318.    }
  319. case'/':
  320. {if(k==0)
  321.    {result=num1/num2;k++;break;}
  322.    else
  323.   { result/=num2;k++;break;}
  324.    }
  325. }
  326. }
  327. if(c=='c')
  328. {
  329. act=0;
  330. num1=str1[0];
  331. num1=str1[8];
  332. memset(str1,0,11);
  333. memset(string1,0,2);
  334. num1=str1[0];
  335. num1=str1[8];
  336. k=0;
  337. }
  338. longdelay(350);
  339. switch(key1)  //显示部分
  340. {case 0x11:   dispcharacter(0,i,'7');i++;break;
  341. case 0x12:    dispcharacter(0,i,'8');i++; break;
  342. case 0x13:    dispcharacter(0,i,'9');i++; break;
  343. case 0x14:    dispcharacter(0,i,'/');i++;break;
  344. case 0x21:    dispcharacter(0,i,'4');i++;break;
  345. case 0x22:    dispcharacter(0,i,'5');i++;break;
  346. case 0x23:    dispcharacter(0,i,'6');i++;break;
  347. case 0x24:    dispcharacter(0,i,'*');i++;break;
  348. case 0x31:    dispcharacter(0,i,'1');i++;break;
  349. case 0x32:    dispcharacter(0,i,'2');i++;break;
  350. case 0x33:    dispcharacter(0,i,'3');i++;break;
  351. case 0x34:    dispcharacter(0,i,'-');i++;break;
  352. case 0x41:    clearscreen();             break;
  353. case 0x42:    dispcharacter(0,i,'0');i++;break;
  354. case 0x43:    dispcharacter(0,i,'=');i++;
  355.                m=result/10000;
  356.                result1=result%10000;
  357.                if(m!=0||result==0)
  358.                {
  359.                n=0;
  360.       m=m+'0';
  361.                dispcharacter(0,i,m);i++;
  362.                }
  363.                m=result1/1000;
  364.                result1=result1%1000;
  365.                if(m!=0||n==0)
  366.         {
  367.                m=m+'0';
  368.       n=0;
  369.                dispcharacter(0,i,m);i++;
  370.       }
  371.                m=result1/100;
  372.                result1=result1%100;
  373.                if(m!=0||n==0)
  374.       {
  375.                m=m+'0';
  376.       n=0;
  377.             dispcharacter(0,i,m);i++;
  378.       }
  379.                m=result1/10;
  380.                result1=result1%10;
  381.                if(m!=0||n==0)
  382.       {
  383.                m=m+'0';
  384.       n=0;
  385.                dispcharacter(0,i,m);i++;
  386.       }
  387.                m=result1;
  388.                if(m!=0||n==0)
  389.                m=m+'0';
  390.                dispcharacter(0,i,m);i++;break;
  391. case 0x44:    dispcharacter(0,i,'+');i++;break;
  392. }
  393. return(i);
  394. }
  395. //---------------------------------------------------------------------------------//
  396. //-------------------------------------主函数---------------------------------------//
  397. void main()
  398. {
  399. char key1,i;
  400. start:initpindata(0);
  401. P2=0X00;
  402. initLCD();
  403. i=0;
  404. scan:key1=key_scan();
  405. i=compute(key1,i);
  406. if(key1==0x41)
  407. goto start;
  408. else
  409. goto scan;
  410. }
复制代码

所有资料51hei提供下载:
简易计算器.zip (207.35 KB, 下载次数: 111)

评分

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

查看全部评分

回复

使用道具 举报

ID:836944 发表于 2020-12-3 16:26 | 显示全部楼层
下下来一试只有加法正常
回复

使用道具 举报

ID:861681 发表于 2020-12-19 09:35 | 显示全部楼层
循环显示555555
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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