找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 6187|回复: 11
收起左侧

如何让单片机控制孵化器温度设定精确到0.1℃

  [复制链接]
ID:229762 发表于 2017-10-14 00:08 | 显示全部楼层 |阅读模式
本人是小白,学单片机就是爱好,从网上找到以下程序源码,上机后发现问题,用按键对温度的加减一次以1℃为单位。不能满足孵化需要。
比如我不能设定温度为38.7℃。
自行改了几次代码都以失败告终,请知道原有的人不吝赐教,先谢过了。
  1. #include<reg52.h>
  2. #include "intrins.h"
  3. typedef unsigned char uint8;
  4. typedef unsigned int uint16;
  5. sbit rs=P2^6;         // 数据命令选择
  6. sbit rw=P2^5;         //读写选择
  7. sbit e=P2^7;          //使能

  8. sbit k1=P1^7; //模式
  9. sbit k2=P2^1; //加
  10. sbit k3=P2^2; //减

  11. sbit DHT11_DQ_OUT=P3^2;
  12. sbit led1=P2^0;         
  13. sbit led2=P3^6;         
  14. sbit led3=P3^5;         
  15. sbit led4=P3^4;         
  16. sbit dq=P3^7;


  17. uint8 mode=0;
  18. float temph=39,templ=37;
  19. char humih=80,humil=20;
  20. char  temp,humi;
  21. uint8 flag;          //设定报警标志
  22. uint8 a,c,tempvalue;
  23. uint8 code num[10]="0123456789";

  24. uint8 code str1[]="Temp:";  //温度
  25. uint8 code str2[]="Humi:";  //湿度
  26. uint8 code str3[]="Error";  
  27. uint8 code str4[]="Success    ";
  28. uint8 code str5[]="%RH";
  29. uint8 code str6[]="TempH:";          //设定温度上限显示
  30. uint8 code str7[]="TempL:";          //设定温度下限显示
  31. uint8 code str8[]="HumiH:";           //设定湿度上限显示
  32. uint8 code str9[]="HumiL:";           //设定湿度下限显示


  33. void delay(uint16 i)
  34. {
  35.         while(i--);
  36. }

  37. void delay_ms(uint16 i)
  38. {
  39.         while(i--)
  40.                 delay(90);
  41. }

  42. void wrc(uint8 c)         //写命令
  43. {
  44.         delay(1000);
  45.         rs=0;
  46.         rw=0;
  47.         e=0;
  48.         P0=c;
  49.         e=1;
  50.         delay(10);
  51.         e=0;
  52. }
  53. void wrd(uint8 dat)          //写数据
  54. {
  55.         delay(1000);
  56.         rs=1;
  57.         rw=0;
  58.         e=0;
  59.         P0=dat;
  60.         e=1;
  61.         delay(10);
  62.         e=0;
  63.         rs=0;
  64. }
  65. void lcd_init()           // LCD1602初始化
  66. {
  67.         delay(1000);
  68.         wrc(0x38);
  69.         wrc(0x38);         //功能设置命令,选择8位总线,双行显示  5*7点阵字符
  70.         wrc(0x38);
  71.         wrc(0x06);        //光标和显示模式设置  光标右移  整屏不移动
  72.         wrc(0x0c);        //显示开关控制  开显示  无光标 光标不闪烁
  73.         wrc(0x01);        //清零指令  固定的
  74. }
  75. //复位DHT11
  76. void DHT11_Rst()           
  77. {                 
  78.     DHT11_DQ_OUT=0;         //拉低DQ
  79.     delay_ms(20);            //拉低至少18ms
  80.     DHT11_DQ_OUT=1;         //DQ=1
  81.         delay(3);             //主机拉高20~40us
  82. }

  83. //等待DHT11的回应
  84. //返回1:未检测到DHT11的存在
  85. //返回0:存在
  86. uint8 DHT11_Check()            
  87. {   
  88.         uint8 retry=0;         
  89.     while (DHT11_DQ_OUT&&retry<100)//DHT11会拉低40~50us
  90.         {
  91.                 retry++;
  92.                 _nop_();
  93.         };         
  94.         if(retry>=100)return 1;
  95.         else retry=0;
  96.     while (!DHT11_DQ_OUT&&retry<100)//DHT11拉低后会再次拉高40~50us
  97.         {
  98.                 retry++;
  99.                 _nop_();
  100.         };
  101.         if(retry>=100)return 1;            
  102.         return 0;
  103. }


  104. //DHT11初始化
  105. //返回0:初始化成功,1:失败
  106. uint8 DHT11_Init()
  107. {
  108.         DHT11_Rst();         
  109.         return DHT11_Check();        
  110. }


  111. //从DHT11读取一个位
  112. //返回值:1/0
  113. uint8 DHT11_Read_Bit(void)                          
  114. {
  115.          uint8 retry=0;
  116.         while(DHT11_DQ_OUT&&retry<100)//等待变为低电平 12-14us 开始
  117.         {
  118.                 retry++;
  119.                 _nop_();
  120.         }
  121.         retry=0;
  122.         while((!DHT11_DQ_OUT)&&retry<100)//等待变高电平         26-28us表示0,116-118us表示1
  123.         {
  124.                 retry++;
  125.                 _nop_();
  126.         }
  127.         delay(1);//等待40us
  128.         if(DHT11_DQ_OUT)return 1;
  129.         else return 0;                  
  130. }

  131. //从DHT11读取一个字节
  132. //返回值:读到的数据
  133. uint8 DHT11_Read_Byte(void)   
  134. {        
  135.     uint8 i,dat=0;
  136.         for (i=0;i<8;i++)
  137.         {
  138.                    dat<<=1;
  139.             dat|=DHT11_Read_Bit();
  140.     }                                                   
  141.     return dat;
  142. }

  143. //从DHT11读取一次数据
  144. //temp:温度值(范围:0~50°)
  145. //humi:湿度值(范围:20%~90%)
  146. //返回值:0,正常;1,读取失败
  147. uint8 DHT11_Read_Data(uint8 *temp,uint8 *humi)   
  148. {        
  149.          uint8 buf[5];
  150.         uint8 i;
  151.         DHT11_Rst();
  152.         if(DHT11_Check()==0)
  153.         {
  154.                 for(i=0;i<5;i++)//读取40位数据
  155.                 {
  156.                         buf[i]=DHT11_Read_Byte();
  157.                 }
  158.                 if((buf[0]+buf[1]+buf[2]+buf[3])==buf[4])
  159.                 {
  160.                         *humi=buf[0];
  161.                         *temp=buf[2];
  162.                 }
  163.                
  164.         }else return 1;
  165.         return 0;            
  166. }




  167. void ds18b20init()                //18b20的初始化
  168. {
  169.         dq=1;
  170.         delay(1);
  171.         dq=0;
  172.         delay(80);
  173.         dq=1;
  174.         delay(5);
  175.         dq=0;
  176.         delay(20);
  177.         dq=1;
  178.         delay(35);
  179. }
  180. void ds18b20wr(uint8 dat)          //18b20写数据
  181. {
  182.         uint8 i;
  183.         for(i=0;i<8;i++)
  184.         {
  185.                 dq=0;
  186.                 dq=dat&0x01;
  187.                 dat>>=1;
  188.                 delay(8);//在时序上只有这一块对时序要求最准确,他的时间必须大于15us
  189.                 dq=1;
  190.                 delay(1);
  191.         }        
  192. }
  193. uint8 ds18b20rd()                  //18b20读数据
  194. {
  195.         uint8 value,i;
  196.         for(i=0;i<8;i++)        
  197.         {
  198.                 dq=0;
  199.                 value>>=1;
  200.                 dq=1;
  201.                 if(dq==1)value|=0x80;
  202.                 delay(8);//在这一块也对时间要求特别准确,整段程序必须大于60us               
  203.         }
  204.         return value;
  205. }
  206. uint8 readtemp()                          //读取温度内需要复位的
  207. {
  208.         uint8 b;
  209.         ds18b20init();                //初始化
  210.         ds18b20wr(0xcc);   //发送忽略ROM指令
  211.         ds18b20wr(0x44);   //发送温度转换指令
  212.         delay(100);
  213.         ds18b20init();           //初始化
  214.         ds18b20wr(0xcc);   //发送忽略ROM指令
  215.         ds18b20wr(0xbe);   //发读暂存器指令
  216.         a=ds18b20rd();         //温度的低八位
  217.         b=ds18b20rd();         //温度的高八位
  218.         b<<=4;                         //ssss s***;s为标志位s=0表示温度值为正数,s=1温度值为负数
  219.         c=b&0x80;                //温度正负标志位确认
  220.         b+=(a&0xf0)>>4;
  221.         a=a&0x0f;          //温度的小数部分
  222.         return b;
  223. }


  224. void key_pros()  //按键处理函数
  225. {
  226.         if(k1==0)
  227.         {
  228.                 delay(1000);
  229.                 if(k1==0)
  230.                 {
  231.                         mode++;
  232.                         if(mode==5)mode=0;
  233.                         wrc(0x01);
  234.                 }
  235.                 while(!k1);
  236.         }
  237.         if(mode==1)                  //对温度上限设定
  238.         {
  239.                 if(k2==0)                  //加
  240.                 {
  241.                         delay(1000);
  242.                         if(k2==0)
  243.                         {
  244.                                 temph++;
  245.                                 if(temph>=80)temph=80;
  246.                         }
  247.                         while(!k2);
  248.                 }
  249.                 if(k3==0)           //减
  250.                 {
  251.                         delay(1000);
  252.                         if(k3==0)
  253.                         {
  254.                                 temph--;
  255.                                 if(temph<=0)temph=0;
  256.                         }
  257.                         while(!k3);
  258.                 }
  259.         }
  260.         if(mode==2)                  //对温度下限设定
  261.         {
  262.                 if(k2==0)                  //加
  263.                 {
  264.                         delay(1000);
  265.                         if(k2==0)
  266.                         {
  267.                                 templ++;
  268.                                 if(templ>=80)templ=80;
  269.                         }
  270.                         while(!k2);
  271.                 }
  272.                 if(k3==0)           //减
  273.                 {
  274.                         delay(1000);
  275.                         if(k3==0)
  276.                         {
  277.                                 templ--;
  278.                                 if(templ<=0)templ=0;
  279.                         }
  280.                         while(!k3);
  281.                 }
  282.         }
  283.         if(mode==3)                  //对湿度上限设定
  284.         {
  285.                 if(k2==0)                  //加
  286.                 {
  287.                         delay(1000);
  288.                         if(k2==0)
  289.                         {
  290.                                 humih++;
  291.                                 if(humih>=80)humih=80;
  292.                         }
  293.                         while(!k2);
  294.                 }
  295.                 if(k3==0)           //减
  296.                 {
  297.                         delay(1000);
  298.                         if(k3==0)
  299.                         {
  300.                                 humih--;
  301.                                 if(humih<=0)humih=0;
  302.                         }
  303.                         while(!k3);
  304.                 }
  305.         }
  306.         if(mode==4)                  //对湿度下限设定
  307.         {
  308.                 if(k2==0)                  //加
  309.                 {
  310.                         delay(1000);
  311.                         if(k2==0)
  312.                         {
  313.                                 humil++;
  314.                                 if(humil>=80)humil=80;
  315.                         }
  316.                         while(!k2);
  317.                 }
  318.                 if(k3==0)           //减
  319.                 {
  320.                         delay(1000);
  321.                         if(k3==0)
  322.                         {
  323.                                 humil--;
  324.                                 if(humil<=0)humil=0;
  325.                         }
  326.                         while(!k3);
  327.                 }
  328.         }
  329. }

  330. void lcd_init_display()           //LCD初始化显示
  331. {
  332.         uint8 i;
  333.         for(i=0;i<5;i++)
  334.         {
  335.                 wrc(0x80+i);
  336.                 wrd(str1[i]);        
  337.         }

  338.         for(i=0;i<5;i++)
  339.         {
  340.                 wrc(0xc0+i);
  341.                 wrd(str2[i]);        
  342.         }
  343. }

  344. void data_pros()        //数据处理函数
  345. {
  346.         uint8 i;              
  347.         uint8 temp_buf[2],humi_buf[2];
  348.         uint8 temphbuf[2],templbuf[2],humihbuf[2],humilbuf[2];
  349.         float dio;
  350.         uint16 k;

  351.         tempvalue=readtemp();
  352.         DHT11_Read_Data(&temp,&humi);
  353.         temp_buf[0]=temp/10+0x30;        
  354.         temp_buf[1]=temp%10+0x30;

  355.         humi_buf[0]=humi/10+0x30;        
  356.         humi_buf[1]=humi%10+0x30;
  357.         
  358.         dio=a*0.0625;
  359.         k=dio*10000;//取小数点后两位有效数字

  360.         temphbuf[0]=temph/10+0x30;
  361.         temphbuf[1]=temph%10+0x30;
  362.         templbuf[0]=templ/10+0x30;
  363.         templbuf[1]=templ%10+0x30;

  364.         humihbuf[0]=humih/10+0x30;
  365.         humihbuf[1]=humih%10+0x30;
  366.         humilbuf[0]=humil/10+0x30;
  367.         humilbuf[1]=humil%10+0x30;

  368.         if(mode==0)
  369.         {
  370.                 lcd_init_display();
  371.                 wrc(0x85);
  372.                 wrd(num[tempvalue%100/10]);
  373.                 wrd(num[tempvalue%100%10]);
  374.                 wrd('.');
  375.                 wrd(num[k/1000]);
  376.                 wrd(0xdf);
  377.                 wrd('C');
  378.         
  379.                 for(i=0;i<2;i++)
  380.                 {
  381.                         wrc(0Xc5+i);
  382.                         wrd(humi_buf[i]);                  
  383.                 }        
  384.                 for(i=0;i<3;i++)
  385.                 {
  386.                         wrc(0Xc7+i);
  387.                         wrd(str5[i]);                  
  388.                 }        
  389.         }
  390.         if(mode==1)                          //温度上限显示
  391.         {
  392.                 wrc(0x80);
  393.                 for(i=0;i<6;i++)
  394.                 {
  395.                         wrd(str6[i]);                  
  396.                 }
  397.                 wrd(temphbuf[0]);
  398.                 wrd(temphbuf[1]);                        
  399.         }
  400.         if(mode==2)                          //温度下限显示
  401.         {
  402.                 wrc(0x80);
  403.                 for(i=0;i<6;i++)
  404.                 {
  405.                         wrd(str7[i]);                  
  406.                 }
  407.                 wrd(templbuf[0]);
  408.                 wrd(templbuf[1]);                        
  409.         }
  410.         if(mode==3)                          //湿度上限显示
  411.         {
  412.                 wrc(0x80);
  413.                 for(i=0;i<6;i++)
  414.                 {
  415.                         wrd(str8[i]);                  
  416.                 }
  417.                 wrd(humihbuf[0]);
  418.                 wrd(humihbuf[1]);                        
  419.         }
  420.         if(mode==4)                          //湿度下限显示
  421.         {
  422.                 wrc(0x80);
  423.                 for(i=0;i<6;i++)
  424.                 {
  425.                         wrd(str9[i]);                  
  426.                 }
  427.                 wrd(humilbuf[0]);
  428.                 wrd(humilbuf[1]);                        
  429.         }
  430. }

  431. void baojinpros()        //报警处理
  432. {
  433.           if(tempvalue>temph)  //jiangwen
  434.                   led1=1;
  435.                   led2=0;
  436.            if(tempvalue<templ) // jiagwen
  437.                   led1=0;
  438.                   led2=1;
  439.                 if(tempvalue>=templ&&tempvalue<=temph)
  440.                 led1=1;
  441.                 led2=1;        

  442.                     if(humi<humil)
  443.                    led3=0;
  444.                    led4=1;
  445.                  if(humi>humih)
  446.                         led3=1;
  447.                    led4=0;
  448.                   if(humi>=humil&&humi<=humih)
  449.                    led3=1;
  450.                    led4=1;

  451. //        if(tempvalue>=temph||humi>=humih)         //检测温度或者湿度高于设定上限值 降温湿
  452. //        {
  453. //                led1=1;                //降温湿指示灯
  454. //                led2=0;
  455. //        }
  456. //        if(tempvalue<=templ||humi<=humil)        //检测温度或者湿度低于设定下限值  升温湿
  457. //        {
  458. //                led1=0;
  459. //                led2=1;           //升高温湿指示灯
  460. //        }
  461. //        if((tempvalue>templ&&tempvalue<temph)&&(humi>humil&&humi<humih))
  462. //        {
  463. //                led1=0;
  464. //                led2=0;        
  465. //        }
  466. }

  467. void main()
  468. {
  469.         uint8 i=0;
  470.         led1=0;
  471.         led2=0;
  472.         lcd_init();
  473.         while(DHT11_Init())        //检测DHT11是否纯在
  474.         {
  475.                 for(i=0;i<5;i++)
  476.                 {
  477.                         wrc(0x80+i);
  478.                         wrd(str3[i]);        
  479.                 }                        
  480.         }
  481.         wrc(0x01);
  482.         lcd_init_display();           //LCD初始化显示
  483.         i=0;
  484.         while(1)
  485.         {        
  486.                 i++;
  487.                 key_pros();
  488.                 baojinpros();        //报警处理
  489.                 if(i==15)
  490.                 {
  491.                         i=0;
  492.                         data_pros();           //读取一次DHT11数据最少要大于100ms
  493.                 }
  494.                 delay(1000);
  495.                
  496.         }        
  497. }
复制代码


回复

使用道具 举报

ID:105206 发表于 2017-10-14 05:49 | 显示全部楼层
你定义的数据类型不对,要设置为浮点型
回复

使用道具 举报

ID:155507 发表于 2017-10-14 08:29 | 显示全部楼层
改个代码


  1. if(mode==1)                  //对温度上限设定
  2.         {
  3.                 if(k2==0)                  //加
  4.                 {
  5.                         delay(1000);
  6.                         if(k2==0)
  7.                         {
  8.                                 temph+=0.1; //temph++; 改
  9.                                 if(temph>=80)temph=80;
  10.                         }
  11.                         while(!k2);
  12.                 }
  13.                 if(k3==0)           //减
  14.                 {
  15.                         delay(1000);
  16.                         if(k3==0)
  17.                         {
  18.                                 temph-=0.1;//temph--;  改
  19.                                 if(temph<=0)temph=0;
  20.                         }
  21.                         while(!k3);
  22.                 }
  23.         }

复制代码
回复

使用道具 举报

ID:158375 发表于 2017-10-14 09:11 | 显示全部楼层
精确到0.1℃,这个基本很难,要是要求准确度0.1℃,更难。
你传感器采用什么?这是个大问题。
回复

使用道具 举报

ID:229762 发表于 2017-10-14 16:42 | 显示全部楼层
传感器采用18b20
回复

使用道具 举报

ID:239426 发表于 2017-10-14 16:59 | 显示全部楼层
很给力
回复

使用道具 举报

ID:123289 发表于 2017-10-14 17:07 | 显示全部楼层
你所采用的传感器必须有0.1度的分辨率。
回复

使用道具 举报

ID:478073 发表于 2019-2-25 03:23 | 显示全部楼层
定义的数据类型不对
回复

使用道具 举报

ID:472640 发表于 2019-2-26 23:56 | 显示全部楼层
db18b20达不到这个精度,要控制到0.1摄氏度,首先要升级传感器,凭借一些常见的传感器和加热装置,肯定不行。还是从数学角度去研究下,比如外界温度变化 加热时间和温升关系,找一些关联的数学模型,另外pid最好也加入其中。弱弱问下,孵化的时候,真要这么精确么?
回复

使用道具 举报

ID:141149 发表于 2019-2-27 00:54 | 显示全部楼层
研究一下
回复

使用道具 举报

ID:521452 发表于 2019-4-25 15:51 | 显示全部楼层
孵化器源码发我一份呗,我刚学习,做个参考。跪求大神了,谢谢。我QQ597904093
回复

使用道具 举报

ID:280876 发表于 2019-4-25 20:45 | 显示全部楼层

db18b20的误差是0.5℃,要精确到0.1要换精度高的传感器。
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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