找回密码
 立即注册

QQ登录

只需一步,快速开始

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

基于51单片机的温度控制源码与仿真

  [复制链接]
跳转到指定楼层
楼主
以AT89C51单片机为核心的控制器,以DS18B20为温度传感器的温度控制器。
首先,通过对元器件的选择,设计控制器的硬件电路;然后,设计相关应用程序;最后,通过仿真,对整个系统进行调试、分析。最终实现温度采集、显示、控制等功能。。

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


单片机源程序如下:
  1. #include<reg51.h>
  2. #include<stdio.h>
  3. #define uchar unsigned char
  4. #define uint unsigned int
  5. sbit ds=P3^2;                                     //温度传感器信号线  
  6. sbit RS=P1^0;
  7. sbit RW=P1^1;
  8. sbit E=P1^2;
  9. sbit beep=P1^5;
  10. sbit man=P1^6;
  11. sbit woman=P1^7;
  12. uint temp;
  13. uint count=0;
  14. uchar f_temp;
  15. unsigned int A,W,C;
  16. data unsigned char SET[15]={"settemp:   "};
  17. data unsigned char REAL[15]={"realtemp:  "};
  18. char code tab[3][4]={  {'1','2','3','4'},
  19.                      {'5','6','7','8'},
  20.                      {'9','0',' ',' '}};       //0到F的16个键植

  21. void delay(uint z)                               //延时函数
  22. { uint x,y;
  23. for (x=z;x>0;x--)
  24.   for (y=110;y>0;y--);
  25. }


  26. //以下是DS18B20的相关程序
  27. void dsreset(void)                             //DS18B20复位初始化函数
  28. { uint i;
  29. ds=0;
  30. i=103;
  31. while(i>0)i--;
  32. ds=1;
  33. i=4;
  34. while(i>0)i--;
  35. }

  36. bit tempreadbit(void)                          //读一位数据函数
  37. { uint i;
  38. bit dat;
  39. ds=0;
  40. i++;                                   //i++起延时作用
  41. ds=1;
  42. i++;
  43. i++;
  44. dat=ds;
  45. i=8;
  46. while(i>0)i--;
  47. return(dat);
  48. }

  49. uchar tempread(void)                         //读1个字节数据函数
  50. { uchar i,j,dat;
  51. dat=0;
  52. for(i=1;i<=8;i++)
  53. { j=tempreadbit();
  54.   dat=(j<<7)|(dat>>1);
  55. }
  56. return(dat);
  57. }

  58. void tempwritebyte(uchar dat)                  //向DS18B20写一个字节数据函数
  59. { uint i;
  60. uchar j;
  61. bit testb;
  62. for (j=1;j<=8;j++)
  63. { testb=dat&0x01;
  64.   dat=dat>>1;
  65.   if (testb)                             //写1
  66.   { ds=0;
  67.    i++;
  68.    i++;
  69.    ds=1;
  70.    i=8;
  71.    while(i>0)i--;
  72.    }
  73. else
  74.    { ds=0;                                //写0
  75.     i=8;
  76.    while(i>0)i--;
  77.    ds=1;
  78.    i++;
  79.    i++;
  80.    }
  81.   }
  82. }

  83. void tempchange(void)                      //DS18B20开始获取温度并进行转换
  84. { dsreset();                   //复位
  85. delay(1);        //延时
  86. tempwritebyte(0xcc);                  //写跳过读ROM指令
  87. tempwritebyte(0x44);                  //写温度转换指令
  88. }

  89. uint get_temp()             //读寄存器中的温度
  90. {  
  91. uchar m,n;               
  92. dsreset();        //复位
  93. delay(5);
  94. tempwritebyte(0xcc);
  95. tempwritebyte(0xbe);
  96. m=tempread();                       //读低8位
  97. n=tempread();                        //读高8位
  98. temp=n;
  99. temp<<=8;
  100. temp=temp|m;
  101. f_temp=temp*0.0625;                                          
  102. temp=f_temp*10+0.5;
  103. f_temp=f_temp+0.05;
  104. return temp;                          //temp是整型
  105. }
  106. //以下是键盘程序
  107. char kbscan()          //键盘扫描
  108. {
  109.   unsigned char hang,lie,key;
  110. P2=0x0f;
  111. if(P2!=0x0f)
  112. delay(1);
  113. if(P2!=0x0f)
  114.   {
  115.   switch(P2&0x0f)
  116.     {
  117.     case 0x0e:lie=0;break;
  118.     case 0x0d:lie=1;break;
  119.     case 0x0b:lie=2;break;
  120.     case 0x07:lie=3;break;
  121.     }
  122.   P2=0xf0;
  123.   P2=0xf0;
  124.   switch(P2&0xf0)
  125.     {
  126.      
  127.     case 0xd0:hang=1;break;
  128.     case 0xb0:hang=2;break;
  129.     case 0xe0:hang=0;break;
  130.     }
  131.   P2=0x0f;
  132.   while(P2!=0x0f);
  133.   key=tab[hang][lie];
  134.   }
  135. else
  136.   key=0;
  137. return (key);
  138. }
  139. //以下是lm016l的显示程序
  140. void cls(unsigned char da)                        //LCD清屏
  141. {
  142.   delay(1);                    //用延时代替检测忙信号
  143.   RS=0;                                                                                
  144.   RW=0;                  
  145.   
  146.   P0=da;                                                                              
  147.   E=1;                                                                                 
  148. E=0;  
  149. }

  150. void write_com(unsigned char c)                  //写入控制命令的子程序
  151. {
  152. int i;
  153. for(i=80;i>0;i--);                           //用延时代替忙检测
  154. RS=0;                                   //选择写命令模式
  155. RW=0;                                  //选择写模式
  156. E=0;
  157. P0=c;                                   //要写的数据
  158. E=1;                                    //给使能端一个高脉冲
  159. E=0;
  160. }

  161. void write_data(unsigned char d)                //写入数据的子程序
  162. {
  163. int i;
  164. for(i=80;i>0;i--);
  165. RS=1;                                  //选择读命令模式
  166. RW=0;                                 //选择写模式
  167. E=0;
  168. P0=d;
  169. E=1;
  170. E=0;
  171. }

  172. void realtemp(te)                                                      
  173. { uchar k;
  174. k=te/10;
  175. REAL[9]=k/10+48;
  176. REAL[10]=k%10+48;
  177. return;
  178. }

  179. void sett()
  180. {   
  181. if(count==1)  
  182.      {  
  183.    SET[13]=kbscan();      
  184.    }
  185. else if(count==2)
  186.   { SET[9]=kbscan();
  187.   }
  188.   else if(count==3)
  189.   { SET[10]=kbscan();
  190.   }
  191.    else if(count==4)
  192.   { SET[12]=kbscan();
  193.   }   
  194. SET[11]='-';
  195. return;
  196. }




  197. //主程序
  198. void main()
  199. {  
  200. int i;
  201. char a,b;
  202.   
  203. write_com(0X01);                     //显示清零数据指针清零
  204. write_com(0X38);                    //设置16*2显示5*7点阵8位数据接口
  205. write_com(0X0C);                    //设置开显示不显示光标
  206. write_com(0X06);                    //写一个字符后地址指针自动加1
  207.   cls(1);              //初始化
  208.   
  209. while(1)
  210. {  tempchange();                    //温度转换函数                                                              
  211.       
  212.   write_com(0x80);                  //显示第一排                                                   
  213.         if(P2!=0x0f)
  214.    {
  215.      if(count==4)                     
  216.       { count=1;}
  217.        
  218.      else count=count+1;
  219.    sett();
  220.     }
  221. for(i=0;i<15;i++)
  222.       {  
  223.    a=SET[i];
  224.         write_data(a);
  225.       }
  226.       
  227.   write_com(0xC0);                      //显示第二排
  228.   realtemp(get_temp());              
  229.      for(i=0;i<15;i++)
  230.        { b=REAL[i];
  231.         write_data(b);      
  232.   }
  233. if(A>C||C>W)
  234. {
  235. beep=1;

  236. }
  237. A=SET[9]*10+SET[10];
  238. W=SET[12]*10+SET[13];
  239. C=REAL[9]*10+REAL[10];


  240. if(A<C&&C<W) {beep=0;man=1;woman=1;}
  241. if(C<W){man=0;}
  242. else if(C>W){man=1;}
  243. if(C>A){woman=0;}
  244. else if(C<A){woman=1;}
  245. }
  246. }
复制代码

所有资料51hei提供下载:
原理图及代码.rar (26.88 KB, 下载次数: 153)


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

使用道具 举报

沙发
ID:312573 发表于 2018-5-2 16:11 | 只看该作者
请问有PID闭环负反馈控制吗?
回复

使用道具 举报

板凳
ID:412814 发表于 2018-10-26 19:39 | 只看该作者
这个真挺好,有源程序,仿真也成功了,慢慢学习了。谢谢
回复

使用道具 举报

地板
ID:418812 发表于 2018-10-31 22:24 | 只看该作者
这个有对应的PCB图吗
回复

使用道具 举报

5#
ID:426681 发表于 2018-11-16 00:18 | 只看该作者
挺好的,感谢
回复

使用道具 举报

6#
ID:432653 发表于 2020-5-22 21:32 | 只看该作者
再加个NO/OFF就更完美了
回复

使用道具 举报

7#
ID:787383 发表于 2020-6-23 19:54 | 只看该作者
这个程序是什么意思
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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