久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 5587|回復: 5
打印 上一主題 下一主題
收起左側

單片機雙路可編程溫度控制系統電路原理圖與源程序等資料

  [復制鏈接]
跳轉到指定樓層
樓主
【簡要說明】
一、尺寸:長72mmX寬99mmX高20mm
二、 主要芯片:單片機DS18B20、數碼管
三、 工作電壓:輸入電壓小于12V,另有24V 可選。功耗小于2W
四、  特點:1、具有輸出電壓指示燈
            2、輸出具有指示燈
            3、采用螺旋壓接端子。
            4、強大的濾波電路。
            5、具有四位數碼管顯示,可以顯示小數點。
            6、具有系統復位功能
            7、具有完善的保護電路:電流限制、熱關斷電路、電源防接反功能、續流保護、光耦隔離等
            8、可接兩個DS18B20傳感器
            9、兩路繼電器獨立工作控制
            10可以自由編程,提供參考程序
            11繼電器所有觸點全部輸出
            12三個輸入控制按鍵,通過程序也可以自由設定
            13工作穩定可靠。           
            14工作溫度范圍-40℃~+70℃
            15、工作濕度 40%  ~ 80%RH
            16、可裝入槽型板,并安裝在DIN導軌上

GYJ-0033_雙路可編程溫度控制系統原理圖及PCB圖


元件清單:


實物圖與接線圖:


A為傳感器1的當前溫度,B為傳感器2的當前溫度,C為設定的差值溫度。

當滿足:(A-B)>=C,時,繼電器吸合,不滿足條件時繼電器斷開。

因為內部是6M晶振  外部是11.0592M  如果選擇外部時鐘,讀取時間達不到60um就不能運行。

兩路溫控兩個探頭溫度差控制單片機源程序如下:
  1. /***********************************************************************
  2. 單品片機;60s2
  3. 板子;     雙路溫控繼電器,兩個溫控探頭,上面的溫控探頭是A面顯示的溫度,  下面的溫控探頭是B面顯示的溫度,
  4. 操作過程; 下完程序先設定C,先按加溫度鍵,然后再按減溫鍵,斷一下電再上電,這樣是為了設定掉電存儲
  5. 板子功能;  當A的溫度大于B的溫度到設定值時,繼電器A吸合,當再這個設定范圍時,斷開。有個問題,當B大于A時也會吸合
  6. *************************************************************************/
  7. #include<reg52.h>
  8. #include<math.h>
  9. #include "INTRINS.H"
  10. #define uchar unsigned char
  11. #define uint  unsigned int
  12. //數碼管顯示段碼
  13. code unsigned char duan[]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0x88, 0x83,  0xC6,  0xBF,0x7f};
  14.                                                                               
  15. unsigned char dong[4] = { 0xFF, 0xFF, 0xFF, 0xFF};        //數碼管顯示緩沖區
  16. uchar i = 0;  //數碼管掃描動態索引
  17. uint time2,time3;
  18. uchar gai = 0;
  19. uchar mode = 1;           //換頁變量

  20. /********************掉電存儲*********************************************/
  21. typedef unsigned char  INT8U;
  22. typedef unsigned int   INT16U;

  23. sfr IAP_DATA    = 0xC2;
  24. sfr IAP_ADDRH   = 0xC3;
  25. sfr IAP_ADDRL   = 0xC4;
  26. sfr IAP_CMD     = 0xC5;
  27. sfr IAP_TRIG    = 0xC6;
  28. sfr IAP_CONTR   = 0xC7;
  29. #define ENABLE_ISP 0x82 //系統工作時鐘<20MHz 時,對IAP_CONTR 寄存器設置此值

  30. union union_temp16
  31. {
  32.     INT16U un_temp16;
  33.     INT8U  un_temp8[2];
  34. }my_unTemp16;

  35. INT8U Byte_Read(INT16U add);              //讀一字節,調用前需打開IAP 功能
  36. void Byte_Program(INT16U add, INT8U ch);  //字節編程,調用前需打開IAP 功能
  37. void Sector_Erase(INT16U add);            //擦除扇區
  38. void IAP_Disable();                       //關閉IAP 功能
  39. void Delay();
  40. /******************************18b20*************************************************************/
  41. bit flag1s = 0;          //1s定時標志

  42. extern bit Start18B20();   //18b20初始化函數
  43. extern bit Get18B20Temp(int *temp);        //18b20溫度讀取函數
  44. /******************************第二路溫控*************************************************/
  45. bit flag1ss = 0;          //1s定時標志

  46. extern bit Start18B200();   //18b20初始化函數
  47. extern bit Get18B20Tempp(int *tempp);        //18b20溫度讀取函數
  48. /*******************************************************************************************/

  49. sbit wei1 = P2^3;         //數碼管的位斷開關
  50. sbit wei2 = P2^4;         
  51. sbit wei3 = P2^5;
  52. sbit wei4 = P2^6;         
  53.          
  54. bit d1 = 1;   //換畫面按鍵當前值
  55. bit d2 = 1;          //計數加按鍵當前值
  56. bit d3 = 1;          //計數減按鍵當前值

  57. sbit s1 =   P2^0;    //計數加
  58. sbit s2 =   P2^1;         //計數減
  59. sbit s3 =   P2^2;         //換畫面按鈕

  60. sbit out1 = P1^2;         //高溫啟動
  61. sbit out2 = P1^3;         //低溫啟動

  62. uchar T0RH = 0;  //T0重載值的高字節
  63. uchar T0RL = 0;  //T0重載值的低字節


  64. void peizhit0(uint ms);        //配置t0定時器
  65. void key();                                //按鍵掃描函數

  66. void main()
  67. {
  68.         bit q1 = 1;
  69.         bit q2 = 1;
  70.         bit q3 = 1;
  71. /***********************18b20***************************************/
  72.         int intT, decT;  //溫度值的整數和小數部分
  73.         bit res ;
  74.     int temp;        //讀取到的當前溫度值
  75.         /***********************第二路18b20***************************************/
  76.         int  intTT, decTT;  //溫度值的整數和小數部分
  77.         bit ress ;
  78.     int tempp;        //讀取到的當前溫度值

  79.         Start18B20(); /*啟動DS18B20*/

  80.         Start18B200(); /*啟動DS18B20*/

  81. /***********************開機讀掉電存儲內容******************************************************/
  82.                 
  83.                
  84.                 time2 = Byte_Read(0x03)*255+Byte_Read(0x02);           //注意這是把高字節和低字節合在一起
  85.                 time3 = Byte_Read(0x05)*255+Byte_Read(0x04);           //讀三的時間
  86.             EA = 1;       //開總中斷
  87.             peizhit0(1);  //配置T0定時1ms

  88.         while(1)
  89.         {
  90. /*********************第一個按鍵換頁按鍵************************************/
  91.                  if(d3 != q3)
  92.                 {
  93.                         q3 = d3;
  94.                         if(d3 == 0)
  95.                         {
  96.                                 mode = mode+1;                                 //功能設置,4個參數,4個周期為一個循環
  97.                                 if(mode == 4)
  98.                                 {
  99.                                         mode = 1;
  100.                                 }
  101.                         }
  102.                 }
  103. /*******************************第二個按鍵按下*************************/

  104.                  if(d2 != q2)
  105.                          {
  106.                                  q2 = d2;
  107.                                 if(d2 == 0)
  108.                                 {
  109.                                   
  110.                                          if(mode ==2)
  111.                                         {
  112.                                                 if(time2>0)
  113.                                                 {
  114.                                                         time2--;
  115.                                                 }
  116.                                         }
  117.                                         else if(mode ==3)
  118.                                         {
  119.                                                 if(time3>0)
  120.                                                 {
  121.                                                         time3--;
  122.                                                 }
  123.                                         }

  124.                                         EA = 0;
  125.                                         Sector_Erase(0);           //擦除0x01地址中的數據
  126.                                        
  127.                                         Byte_Program(0x02,time2);
  128.                                         Byte_Program(0x03,time2>>8);
  129.                                         Byte_Program(0x04,time3);
  130.                                         Byte_Program(0x05,time3>>8);
  131.                                         EA = 1;
  132.                                  }
  133.                            }
  134. /*****************************第二個按鍵按下***************************/

  135.                  if(d1 != q1)
  136.                          {
  137.                                  q1 = d1;
  138.                                 if(d1 == 0)
  139.                                 {
  140.                                   
  141.                                          if(mode ==2)                                          // b
  142.                                         {
  143.                                                 time2 = (time2+1)%999;                          
  144.                                         }
  145.                                         else if(mode ==3)
  146.                                         {
  147.                                                 time3 = (time3+1)%999;//c                          
  148.                                         }

  149.                                         EA = 0;
  150.                                         Sector_Erase(0);           //擦除0x01地址中的數據
  151.                                        
  152.                                         Byte_Program(0x02,time2);
  153.                                         Byte_Program(0x03,time2>>8);
  154.                                         Byte_Program(0x04,time3);
  155.                                         Byte_Program(0x05,time3>>8);
  156.                                         EA = 1;
  157.                                  }
  158.                            }

  159.   /***************第一層顯示**************************/
  160.                            if(mode == 1)                                          
  161.                         {
  162.                                  
  163.                                 dong[0] = duan [10];
  164.                                 dong[1] = duan [intT/100%10];
  165.                         dong[2] = duan [intT/10%10];
  166.                         dong[3] = duan [intT%10];
  167.                         }
  168.                   /*************第二層顯示**************************/
  169.                            if(mode == 2)                                          
  170.                         {
  171.                                 dong[0] = duan [11];
  172.                                 dong[1] = duan [intTT/100%10];
  173.                         dong[2] = duan [intTT/10%10];
  174.                         dong[3] = duan [intTT%10];
  175.                         }
  176.                  /*************第三層顯示**************************/
  177.                            if(mode == 3)                                          
  178.                         {
  179.                                 dong[0] = duan [12];
  180.                                 dong[1] = duan [time3/100%10];
  181.                         dong[2] = duan [time3/10%10];
  182.                         dong[3] = duan [time3%10];
  183.                         }
  184. /*****************************溫控部分**************************************************/

  185.             if (flag1s)  //每秒更新一次溫度
  186.         {
  187.                           flag1s = 0;
  188.                              gai++;
  189.                           Start18B20(); // 注意  一定要隨著溫度讀取函數一起每秒更新啟動一次 不然就只能讀取到剛上電那一瞬間的溫度   啟動DS18B20
  190.                   res = Get18B20Temp(&temp);  //讀取當前溫度

  191.                   intT = temp*10 >> 4;             // 注意在這個地方temp*10就是精確1位小數點,*100就是精確兩位小數點 但是只能精確一位小數點   分離出溫度值整數部分
  192.             decT = temp & 0xF;            //分離出溫度值小數部分
  193.           /*
  194.                          if((intT <= time2) && (intT >= time3))                         //注意  控制部分要放到  這個函數內 不然上電就會先比較  會有動作  放在這里就可以先讀取再比較  穩定
  195.                  {
  196.                          out2 = 1;
  197.                         out1 = 1;       
  198.                  }
  199.                  if(intT >= time2)
  200.                  {
  201.                          out2 = 1;
  202.                         //out1 = 0;
  203.                         if(gai >= 3)
  204.                         {
  205.                                 gai = 0;
  206.                                 out1 = ~out1;       
  207.                         }
  208.                  }
  209.                  
  210.                  if(intT <= time3)
  211.                  {
  212.                          out1 = 1;
  213.                         //out2 = 0;
  214.                                 if(gai >= 3)
  215.                         {
  216.                                 gai = 0;
  217.                                 out2 = ~out2;       
  218.                         }
  219.        
  220.                  }
  221.     */
  222.           }
  223.                
  224.           
  225.           if (flag1ss)  //每秒更新一次溫度
  226.         {
  227.                           flag1ss = 0;
  228.                              
  229.                           Start18B200(); // 注意  一定要隨著溫度讀取函數一起每秒更新啟動一次 不然就只能讀取到剛上電那一瞬間的溫度   啟動DS18B20
  230.                   ress = Get18B20Tempp(&tempp);  //讀取當前溫度

  231.                   intTT = tempp*10 >> 4;             // 注意在這個地方temp*10就是精確1位小數點,*100就是精確兩位小數點 但是只能精確一位小數點   分離出溫度值整數部分
  232.             decTT = tempp & 0xF;            //分離出溫度值小數部分


  233.                          
  234.       }
  235.         /*************************************************************/
  236.         if((intT-intTT) >= time3)
  237.         {
  238.                 out2 = 0;
  239.         }
  240.         else
  241.         {
  242.                 out2 = 1;
  243.         }
  244.                        
  245. /************************************************************************/
  246.         }       
  247. }

  248. /* 配置并啟動T0,ms-T0定時時間 */
  249. void peizhit0(uint ms)
  250. {
  251.     unsigned long tmp;  //臨時變量
  252.    
  253.     tmp = 11059200 / 12;      //定時器計數頻率           注意 因為晶振是11.0592,,12個震蕩周期才是一個機器周期,所以,計數器加一所用的頻率就是11059200/12
  254.     tmp = (tmp * ms) / 1000;  //計算所需的計數值   注意 上面的計數時間單位是秒,所以除以1000就轉化為ms了
  255.     tmp = 65536 - tmp;        //計算定時器重載值
  256.     tmp = tmp + 18;           //補償中斷響應延時造成的誤差
  257.     T0RH = (unsigned char)(tmp>>8);  //定時器重載值拆分為高低字節        注意  因為是char型所以這個數據如果不向左移動8位他就只能保存低位的8位數據,一個char型變量保存是從低8位先保存,保存完后如果有空間再保存高位,向右移動8位就是讓它從高位開始保存,這個16位計數換成二進制是 1111 1000 1101 1110
  258.     T0RL = (unsigned char)tmp;                 //直接保存低字節數據
  259.     TMOD &= 0xF0;   //清零T0的控制位
  260.     TMOD |= 0x01;   //配置T0為模式1
  261.     TH0 = T0RH;     //加載T0重載值
  262.     TL0 = T0RL;
  263.     ET0 = 1;        //使能T0中斷
  264.     TR0 = 1;        //啟動T0
  265. }
  266. /*按鍵掃描函數*/
  267. void key()
  268. {
  269.         static uchar saomiaozhi[3] = {1,1,1};
  270.         saomiaozhi[0] = (saomiaozhi[0]<<1) | s1;
  271.         saomiaozhi[1] = (saomiaozhi[1]<<1) | s2;
  272.         saomiaozhi[2] = (saomiaozhi[2]<<1) | s3;

  273.         if(saomiaozhi[0] == 0x00)
  274.         {
  275.                 d1 = 0;
  276.         }
  277.         if(saomiaozhi[0] == 0xff)
  278.         {
  279.                 d1 = 1;
  280.         }
  281.         if(saomiaozhi[1] == 0x00)
  282.         {
  283.                 d2 = 0;
  284.         }
  285.         if(saomiaozhi[1] == 0xff)
  286.         {
  287.                 d2 = 1;
  288.         }
  289.         if(saomiaozhi[2] == 0x00)
  290.         {
  291.                 d3 = 0;
  292.         }
  293.         if(saomiaozhi[2] == 0xff)
  294.         {
  295.                 d3 = 1;
  296.         }
  297.        
  298. }

  299. /* T0中斷服務函數,完成數碼管、按鍵掃描與秒表計數 */
  300. void t0() interrupt 1
  301. {
  302.         static uchar c = 0;
  303.         static unsigned int tmr1s = 0;
  304.         static unsigned int tmr1ss = 0;
  305.     TH0 = T0RH;  //重新加載重載值
  306.     TL0 = T0RL;
  307.         c++;
  308.         tmr1s++;
  309.         tmr1ss++;
  310.         if (tmr1s >= 1000)  //定時1s
  311.     {
  312.         tmr1s = 0;
  313.         flag1s = 1;
  314.                  
  315.     }
  316.         if (tmr1ss >= 1000)  //定時1s
  317.     {
  318.         tmr1ss = 0;
  319.         flag1ss = 1;
  320.                  
  321.     }
  322.         if(c >= 2)
  323.         {  c = 0;
  324.            key();                 //按鍵掃描函數
  325.         }

  326.         P0 = 0xff;
  327.         switch (i)
  328.    {
  329.              case 0: wei1 = 0; wei2 = 1; wei3 = 1;wei4 = 1; i++;  P0 = dong[0];     break;
  330.           case 1: wei1 = 1; wei2 = 0; wei3 = 1;wei4 = 1; i++;  P0 = dong[1];     break;
  331.           case 2: wei1 = 1; wei2 = 1; wei3 = 0;wei4 = 1; i++;  P0 = dong[2];     break;
  332.           case 3: wei1 = 1; wei2 = 1; wei3 = 0;wei4 = 1; i++;  P0 = 0x7f;     break;
  333.           case 4: wei1 = 1; wei2 = 1; wei3 = 1;wei4 = 0; i=0;  P0 = dong[3];     break;
  334.           default: break;
  335.         }
  336. }
  337. /******************************掉電儲存功能********************************************************/
  338. //讀一字節,調用前需打開IAP 功能,入口:DPTR = 字節地址,返回:A = 讀出字節
  339. INT8U Byte_Read(INT16U add)
  340. {
  341.     IAP_DATA = 0x00;
  342.     IAP_CONTR = ENABLE_ISP;         //打開IAP 功能, 設置Flash 操作等待時間
  343.     IAP_CMD = 0x01;                 //IAP/ISP/EEPROM 字節讀命令

  344.     my_unTemp16.un_temp16 = add;
  345.     IAP_ADDRH = my_unTemp16.un_temp8[0];    //設置目標單元地址的高8 位地址
  346.     IAP_ADDRL = my_unTemp16.un_temp8[1];    //設置目標單元地址的低8 位地址

  347.     //EA = 0;
  348.     IAP_TRIG = 0x5A;   //先送 5Ah,再送A5h 到ISP/IAP 觸發寄存器,每次都需如此
  349.     IAP_TRIG = 0xA5;   //送完A5h 后,ISP/IAP 命令立即被觸發起動
  350.     _nop_();
  351.     //EA = 1;
  352.     IAP_Disable();  //關閉IAP 功能, 清相關的特殊功能寄存器,使CPU 處于安全狀態,
  353.                     //一次連續的IAP 操作完成之后建議關閉IAP 功能,不需要每次都關
  354.     return (IAP_DATA);
  355. }

  356. //字節編程,調用前需打開IAP 功能,入口:DPTR = 字節地址, A= 須編程字節的數據
  357. void Byte_Program(INT16U add, INT8U ch)
  358. {
  359.     IAP_CONTR = ENABLE_ISP;         //打開 IAP 功能, 設置Flash 操作等待時間
  360.     IAP_CMD = 0x02;                 //IAP/ISP/EEPROM 字節編程命令

  361.     my_unTemp16.un_temp16 = add;
  362.     IAP_ADDRH = my_unTemp16.un_temp8[0];    //設置目標單元地址的高8 位地址
  363.     IAP_ADDRL = my_unTemp16.un_temp8[1];    //設置目標單元地址的低8 位地址

  364.     IAP_DATA = ch;                  //要編程的數據先送進IAP_DATA 寄存器
  365.     //EA = 0;
  366.     IAP_TRIG = 0x5A;   //先送 5Ah,再送A5h 到ISP/IAP 觸發寄存器,每次都需如此
  367.     IAP_TRIG = 0xA5;   //送完A5h 后,ISP/IAP 命令立即被觸發起動
  368.     _nop_();
  369.     //EA = 1;
  370.     IAP_Disable();  //關閉IAP 功能, 清相關的特殊功能寄存器,使CPU 處于安全狀態,
  371.                     //一次連續的IAP 操作完成之后建議關閉IAP 功能,不需要每次都關
  372. }

  373. //擦除扇區, 入口:DPTR = 扇區地址
  374. void Sector_Erase(INT16U add)
  375. {
  376.     IAP_CONTR = ENABLE_ISP;         //打開IAP 功能, 設置Flash 操作等待時間
  377.     IAP_CMD = 0x03;                 //IAP/ISP/EEPROM 扇區擦除命令

  378.     my_unTemp16.un_temp16 = add;
  379.     IAP_ADDRH = my_unTemp16.un_temp8[0];    //設置目標單元地址的高8 位地址
  380.     IAP_ADDRL = my_unTemp16.un_temp8[1];    //設置目標單元地址的低8 位地址

  381.     //EA = 0;
  382.     IAP_TRIG = 0x5A;   //先送 5Ah,再送A5h 到ISP/IAP 觸發寄存器,每次都需如此
  383.     IAP_TRIG = 0xA5;   //送完A5h 后,ISP/IAP 命令立即被觸發起動
  384.     _nop_();
  385.     //EA = 1;
  386.     IAP_Disable();  //關閉IAP 功能, 清相關的特殊功能寄存器,使CPU 處于安全狀態,
  387.                     //一次連續的IAP 操作完成之后建議關閉IAP 功能,不需要每次都關
  388. }

  389. void IAP_Disable()
  390. {
  391.     //關閉IAP 功能, 清相關的特殊功能寄存器,使CPU 處于安全狀態,
  392.     //一次連續的IAP 操作完成之后建議關閉IAP 功能,不需要每次都關
  393.     IAP_CONTR = 0;      //關閉IAP 功能
  394.     IAP_CMD   = 0;      //清命令寄存器,使命令寄存器無命令,此句可不用
  395.     IAP_TRIG  = 0;      //清命令觸發寄存器,使命令觸發寄存器無觸發,此句可不用
  396.     IAP_ADDRH = 0;
  397.     IAP_ADDRL = 0;
  398. ……………………

  399. …………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼

所有資料51hei提供下載:
GYJ-0033_雙路可編程溫度控制系統發貨資料.rar (5.06 MB, 下載次數: 85)


分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏10 分享淘帖 頂 踩
回復

使用道具 舉報

沙發
ID:483092 發表于 2019-3-2 09:59 | 只看該作者
寫得非常好
回復

使用道具 舉報

板凳
ID:333807 發表于 2019-4-5 11:58 | 只看該作者

寫得非常好
回復

使用道具 舉報

地板
ID:630324 發表于 2019-10-26 08:40 | 只看該作者
寫的好,學習了
回復

使用道具 舉報

5#
ID:282431 發表于 2019-10-26 09:39 | 只看該作者
資料非常好,謝謝樓主
回復

使用道具 舉報

6#
ID:588724 發表于 2020-1-27 00:08 | 只看該作者
樓主是職場專業的工程師吧
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 欧美日韩精品一区二区三区四区 | 国产男女视频 | 日操夜操 | 男女黄网站 | 久久久国产精品视频 | 精品三级在线观看 | 91精品国产一二三 | 99精品国产在热久久 | 国产精品国产三级国产aⅴ浪潮 | 男女久久久 | 国产又爽又黄的视频 | 一区二区精品视频 | 精品一区二区三区91 | 中文字幕在线免费 | 在线看av网址 | 国产精彩视频 | 国产精品九九九 | 一级黄色裸片 | 中文字幕在线播放不卡 | 久久精品福利视频 | 女人精96xxx免费网站p | 国产美女特级嫩嫩嫩bbb片 | 免费日韩av网站 | xxxxxx国产 | 久久久久久久久一区 | 久久中文字幕电影 | 人人性人人性碰国产 | 久久爱黑人激情av摘花 | 一区二区三区精品 | 国产精品三级 | 18性欧美 | 国产第一区二区 | 男女羞羞视频网站 | 欧美一区二区在线看 | 国产99精品| www.久| 中文字幕在线精品 | 成人在线视频观看 | 99自拍视频 | 午夜久草 | 999热在线视频 |