找回密码
 立即注册

QQ登录

只需一步,快速开始

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

基于c51的简易计算器的制作(含c语言代码与Proteus原理图)

  [复制链接]
跳转到指定楼层
楼主
基于4*4矩阵,89c51,lcd1602显示器
附件中内容如下:原理图,hex文件,.c文件(源码),仿真程序
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)

代码如下:
  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, 下载次数: 278)


评分

参与人数 2黑币 +55 收起 理由
51单片机1234 + 5
admin + 50 共享资料的黑币奖励!

查看全部评分

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

使用道具 举报

沙发
ID:350833 发表于 2018-6-14 16:22 | 只看该作者
欢迎各位参观学习!!!
回复

使用道具 举报

板凳
ID:354047 发表于 2018-6-19 10:08 | 只看该作者
楼主 这个程序能在15系列上实现吗
回复

使用道具 举报

地板
ID:362129 发表于 2018-6-30 09:39 | 只看该作者
学习一下
回复

使用道具 举报

5#
ID:362129 发表于 2018-6-30 10:37 | 只看该作者
这个可以用吗
回复

使用道具 举报

6#
ID:362129 发表于 2018-6-30 10:39 | 只看该作者
这个里面是直接有proteus的仿真文件是吗 ?就直接可以运行?
回复

使用道具 举报

7#
ID:362129 发表于 2018-6-30 10:41 | 只看该作者
看到的话可以加qq联系一下吗?1347634544
回复

使用道具 举报

8#
ID:392176 发表于 2018-8-31 23:38 | 只看该作者
参观学习
回复

使用道具 举报

9#
ID:393375 发表于 2018-9-7 15:29 来自手机 | 只看该作者
小白一个,学习学习
回复

使用道具 举报

10#
ID:393375 发表于 2018-9-12 09:35 来自手机 | 只看该作者
请问这个程序能在单片机开发板上实现功能嘛
回复

使用道具 举报

11#
ID:396896 发表于 2018-9-12 11:04 | 只看该作者
学习了,谢谢
回复

使用道具 举报

12#
ID:396896 发表于 2018-9-12 11:29 | 只看该作者
谢谢,我要好好学习一下
回复

使用道具 举报

13#
ID:308173 发表于 2018-9-12 12:14 | 只看该作者
为什么KEIL打不开呢
回复

使用道具 举报

14#
ID:399809 发表于 2018-9-19 08:56 | 只看该作者
不错,感谢楼主分享
回复

使用道具 举报

15#
ID:338810 发表于 2018-9-27 16:28 | 只看该作者

楼主 这个程序能在15系列上实现吗
回复

使用道具 举报

16#
ID:92810 发表于 2018-10-9 23:08 | 只看该作者
感觉挺不错的样子,可以下载学习
回复

使用道具 举报

17#
ID:229361 发表于 2018-10-10 00:54 | 只看该作者

请问这个程序能在单片机开发板上实现功能嘛
回复

使用道具 举报

18#
ID:403554 发表于 2018-10-10 09:36 | 只看该作者
如何把它改为4*6的键盘呢
回复

使用道具 举报

19#
ID:116400 发表于 2018-12-23 15:22 | 只看该作者
你这个程序是不可以的,有些简单的数值是算不出来的,总是出错你看看看,用你的程序用你的仿真源文件计算的结果
回复

使用道具 举报

20#
ID:450444 发表于 2018-12-25 12:48 | 只看该作者
WANGCHONG 发表于 2018-12-23 15:22
你这个程序是不可以的,有些简单的数值是算不出来的,总是出错你看看看,用你的程序用你的仿真源文件计算的 ...

你说的都对
回复

使用道具 举报

21#
ID:462368 发表于 2019-1-5 11:54 | 只看该作者
good
回复

使用道具 举报

22#
ID:558095 发表于 2019-6-9 16:59 | 只看该作者


楼主 这个程序能在C52单片机上实现吗
回复

使用道具 举报

23#
ID:558629 发表于 2019-6-10 08:46 | 只看该作者
真好真好真好
回复

使用道具 举报

24#
ID:496206 发表于 2019-6-12 09:48 | 只看该作者
keil根本打不开
回复

使用道具 举报

25#
ID:577482 发表于 2019-7-3 22:35 | 只看该作者
学习一下
回复

使用道具 举报

26#
ID:654919 发表于 2019-12-3 10:33 | 只看该作者
厉害了
回复

使用道具 举报

27#
ID:665084 发表于 2019-12-16 21:17 | 只看该作者
哇,学习一下
回复

使用道具 举报

28#
ID:655938 发表于 2019-12-31 20:27 | 只看该作者
参观学习
回复

使用道具 举报

29#
ID:730070 发表于 2020-5-10 01:50 来自手机 | 只看该作者
51单片机也能实现大部分功能,ARM虽然高效,很多用51和PIC单片机也够了
回复

使用道具 举报

30#
ID:753881 发表于 2020-5-17 11:30 | 只看该作者
很好用很方便
回复

使用道具 举报

31#
ID:631764 发表于 2020-5-17 12:24 | 只看该作者
参观学习,感谢楼主
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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