亚洲春色中文字幕久久久-三上亚,一吻二脱三床四吻胸,国产真实伦对白视频全集,在线毛片观看,精品成品入口黄网,国产毛aⅴ片久久久,亚洲AV色香蕉一区二区三区老师,萧皇后A级艳片,色情日本视频更新,99久久亚洲精品日本无码

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

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

簡單51單片機多點18b20溫度測量系統源碼

[復制鏈接]
跳轉到指定樓層
樓主
ID:280461 發表于 2018-1-28 18:21 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
分享我的期末作業,基于51單片機的多點溫度測量
同時多點溫度測量是在單總線上掛載多個18b20溫度傳感器

單片機源程序如下:
  1. #include <reg51.h>      
  2. #include <intrins.h>  
  3. #define uchar unsigned char  
  4. #define uint unsigned int  
  5. #define MAXNUM 4        //宏定義單總線上最大可掃描DS18B20個數  
  6. //*****************************//
  7. //**初始定義管腳、變量與數組***//
  8. //*****************************//     
  9. sbit DS=P3^7;
  10. sbit lcden=P2^7;//液晶使能端
  11. sbit lcdrs=P2^6;//液晶數0據命令選擇端
  12. sbit lcdrw=P2^5;
  13. union{                 
  14.     uchar c[2];        
  15.     uint x;            
  16. }temp;                                          
  17. uchar time=0;   
  18. uint cc,xs;             //變量cc中保存計算出的溫度值的整數部分,xs保存計算出的溫度值的小數部分的第一位            
  19. uchar idata disbuffer[6];   //LCD顯示緩存數組  
  20. uchar idata ID[4][8]={0};//{{0x28,0xff,0x80,0x2e,0x70,0x16,0x03,0xad},{0x28,0xff,0x95,0xb5,0x81,0x16,0x03,0x50},{0x28,0xff,0x02,0x96,0xa1,0x16,0x04,0x59},{0x28,0xff,0xf0,0xf5,0x62,0x16,0x04,0xd3}};
  21.                        //{"82FF08E2076130DA","82FF595B18613005","82FF20691A614095","82FF0F5F2661403D"};   //用于記錄各DS18B20的ROM序列號  
  22. uchar idata RomID_temp[8];  //匹配DS18B20時臨時記錄要匹配DS18B20的序列號
  23. uchar east[5]=" EAST";
  24. uchar west[5]=" WEST";
  25. uchar south[5]="SOUTH";
  26. uchar north[5]="NORTH";
  27. uchar m=0;            
  28. uchar num=0;           
  29. //***************************//
  30. //*******18b20時序延時*******//
  31. //***************************//
  32. void delay(uint i)      //i*9.62us   
  33. {  
  34.     uint j;  
  35.     for(j=i;j>0;j--);  
  36. }  
  37.   
  38. void delay_ms(uchar i)  //(j*2+1+2)*i+5     
  39. { uchar j;              //12MHz   0.5*i ms  
  40.   do{j=248;            
  41.      do{j--;}while(j);   
  42.      i--;   
  43.     }while(i);   
  44. }  
  45.    
  46. void delay_2us(uchar i)   // 2*i+5 us  
  47. {  
  48.   while(--i);  
  49. }  
  50. //**************************//
  51. //******18b20子程***********//
  52. //**************************//  
  53. uchar DS_init(void)       //18B20復位,初始化函數  
  54. {  
  55.   uchar presence;  
  56.   DS=0;          delay_2us(250); //根據DS18B20的復位時序.先把總線拉低555us  
  57.   DS=1;          delay_2us(30);  //再釋放總線,65us后讀取DS18B20發出的信號  
  58.   presence=DS;   delay_2us(250); //如果復位成功,則presence的值為0;否則為1  
  59.   return (presence);             //返回0則初始化成功,否則失敗  
  60. }   
  61.   
  62. uchar read_byte(void)       //讀1字節  
  63. {  
  64.   uchar i,j,dat=0;  
  65.   for(i=1;i<=8;i++)          //作8個循環,讀出的8位組成一個字節  
  66.     {DS=0;   _nop_();       //先將總線拉低1us,   
  67.      DS=1;   delay_2us(2);  //再釋放總線,產生讀起始信號,延遲9us后讀取總線上的DS18B20發出的值  
  68.      j=DS;   delay_2us(30); //一位讀完后,延遲65us后讀下一位  
  69.      dat=(j<<7)|(dat>>1);   //讀出的數據最低位在一個字節的最低位,這樣剛好一個字節在DAT里  
  70.     }  
  71.   return(dat);  
  72. }   
  73.   
  74. uchar read_2bit(void)       //讀2位  
  75. {  
  76.   uchar i=0,j=0;  
  77.   DS=0;   _nop_();          //先將總線拉低1us,   
  78.   DS=1;   delay_2us(2);     //再釋放總線,產生讀起始信號,延遲9us后讀取總線上的DS18B20發出的值  
  79.   j=DS;   delay_2us(30);    //一位讀完后,延遲65us后讀下一位  
  80.   DS=0;   _nop_();            
  81.   DS=1;   delay_2us(2);  
  82.   i=DS;   delay_2us(30);  
  83.   i=j*2+i;              //將讀出的兩位放到變量i中,其中第一個讀出的位處于i的第1位;而第二個讀出的位處于i的第0位  
  84.   return(i);  
  85. }   
  86.    
  87. void write_byte(uchar dat)  //寫1字節  
  88. {   
  89.   uchar i;  
  90.   for(i=0;i<8;i++)           //作8個循環,寫入的8位組成一個字節  
  91.     {DS=0;                  //先將總線拉低  
  92.      DS = dat&0x01;         //向總線上放入要寫的值  
  93.      delay_2us(50);         //延遲105us,以使DS18B20能采樣到要寫入的值  
  94.      DS = 1;                //釋放總線,準備寫入下一位  
  95.      dat>>=1;             //將要寫的下一位移到dat的最低位        
  96.     }  
  97. }   
  98.    
  99. void write_bit(bit dat) //寫1位  
  100. {   
  101.   DS=0;             //先將總線拉低  
  102.   DS=dat;           //向總線上放入要寫的值  
  103.   delay_2us(50);    //延遲105us,以使DS18B20能采樣到要寫入的值  
  104.   DS = 1;           //釋放總線  
  105. }  
  106. //************************//
  107. //*******1602延時*********//
  108. //************************//
  109. void delay1(uint z)
  110. {
  111.         uint i,j;
  112.         for(i=z;i>0;i--)
  113.         for(j=110;j>0;j--);
  114. }
  115. //************************//
  116. //******1602子程**********//
  117. //************************//
  118. void write_com(uchar com) //寫命令
  119. {
  120.         lcdrs=0;//選擇寫命令模式
  121.         P0=com;//將要寫的命令字送到數據總線上
  122.         delay1(5);//稍作延時以待數據穩定
  123.         lcden=1;//使能端給一高電平脈沖,因為初始化函數中已將lcden置零
  124.         delay1(5);//稍作延時
  125.         lcden=0;//將使能端置零完成高脈沖
  126. }

  127. void write_data(uchar date)  //寫數據
  128. {
  129.         lcdrs=1;//選擇寫數據操作
  130.         P0=date;//將要寫的數據送到數據總線上
  131.         delay1(5);//稍作延時
  132.         lcden=1;//使能端給一高電平脈沖
  133.         delay1(5);
  134.         lcden=0;
  135. }

  136. void init()   //1602初始化
  137. {
  138.         lcdrw=0;
  139.         lcden=0;
  140.         write_com(0x38);//設置16*2顯示,5*7點陣,8位數據接口
  141.         write_com(0x0c);//設置開顯示,不顯示光標
  142.         write_com(0x06);//寫一個字節后地址指針加1
  143.         write_com(0x01);//顯示清零,數據指針清零
  144.         write_com(0x80);//設置數據指針起點
  145. }
  146. //****************************//
  147. //***讀取18b20溫度數據子程 ***//
  148. //****************************//
  149. void Read_Temperature_rom(void) //讀取溫度函數  
  150. {   uchar i;  
  151.     DS_init();  
  152.     write_byte(0x55);       //匹配ROM  
  153.     for(i=0;i<8;i++)     //發出64位ROM編碼  
  154.             write_byte(RomID_temp[i]);  
  155.     write_byte(0x44);       //開始轉換溫度
  156.     DS_init();  
  157.     write_byte(0x55);       //匹配ROM  
  158.     for(i=0;i<8;i++)     //發出64位ROM編碼  
  159.             write_byte(RomID_temp[i]);  
  160.     write_byte(0xBE);       //發讀溫度命令  
  161.     temp.c[1]=read_byte();  //讀低字節,之所以c[1]中放低字節,是因為C51采用的是大端格式
  162.     temp.c[0]=read_byte();  //讀高字節,之所以c[0]中放低字節,是因為C51采用的是大端格式
  163.                             //共用體定義
  164. }  
  165.   
  166. void Temperature_cov(void)      //溫度轉化
  167. {                              
  168.     cc=temp.x/16;           //計算出溫度值的整數部分,這個語句相當于數值乘0.0625再取整數部分  
  169.     xs=temp.x&0x0f;         //取溫度值小數部分的第一位  
  170.     xs=xs*10;               //這兩條語句相當于乘0.625,得小數位的第一位,注意不是乘0.0625  
  171.     xs=xs/16;
  172. }  
  173. //***********************//
  174. //******顯示子程*********//
  175. //***********************//
  176. void display_m(void)   //顯示18b20個數或方向
  177. {
  178.         uchar i;
  179.         write_com(0x8B);
  180.         //write_data('0'+m);
  181.         if(m==0)
  182.         for(i=0;i<5;i++)
  183.         {write_data(east[i]);}
  184.         if(m==1)
  185.         for(i=0;i<5;i++)
  186.         {write_data(west[i]);}
  187.         if(m==2)
  188.         for(i=0;i<5;i++)
  189.         {write_data(south[i]);}
  190.         if(m==3)
  191.         for(i=0;i<5;i++)
  192.         {write_data(north[i]);}
  193. }

  194. void display_ROMID(void)        //顯示序列號  
  195. {   
  196.     uchar tmp[]={0x30,0x31,0x32,0x33,0x34,0x35,0x36,
  197.              0x37,0x38,0x39,0x41,0x42,0x43,0x44,0x45,0x46};       
  198.           uchar k,i;
  199.     uchar disbuffer_rom[16];                                                        
  200.              disbuffer_rom[0]=(ID[m][k]&0x0F);  
  201.              disbuffer_rom[1]=((ID[m][k]&0xF0)>>4);  
  202.              disbuffer_rom[2]=(ID[m][k+1]&0x0F);  
  203.              disbuffer_rom[3]=((ID[m][k+1]&0xF0)>>4);  
  204.              disbuffer_rom[4]=(ID[m][k+2]&0x0F);  
  205.              disbuffer_rom[5]=((ID[m][k+2]&0xF0)>>4);  
  206.              disbuffer_rom[6]=(ID[m][k+3]&0x0F);  
  207.              disbuffer_rom[7]=((ID[m][k+3]&0xF0)>>4);
  208.              disbuffer_rom[8]=(ID[m][k+4]&0x0F);  
  209.              disbuffer_rom[9]=((ID[m][k+4]&0xF0)>>4);
  210.              disbuffer_rom[10]=(ID[m][k+5]&0x0F);  
  211.              disbuffer_rom[11]=((ID[m][k+5]&0xF0)>>4);
  212.              disbuffer_rom[12]=(ID[m][k+6]&0x0F);  
  213.              disbuffer_rom[13]=((ID[m][k+6]&0xF0)>>4);
  214.              disbuffer_rom[14]=(ID[m][k+7]&0x0F);  
  215.              disbuffer_rom[15]=((ID[m][k+7]&0xF0)>>4);                                         
  216.                  write_com(0x80+0x40);
  217.     for(i=0;i<16;i++)               
  218.                         {
  219.                                
  220.                           write_data(tmp[disbuffer_rom[i]]);    //1602寫入序列號
  221.                         }         
  222. }  
  223.    
  224. void display(void)         //溫度顯示
  225. {  
  226.         disbuffer[0]='+';
  227.         disbuffer[1]=cc/10;
  228.         disbuffer[2]=cc%10;
  229.         disbuffer[3]='.';
  230.         disbuffer[4]=xs;
  231.         disbuffer[5]='C';
  232.         write_com(0x80);
  233.   write_data(disbuffer[0]);
  234.         write_com(0x81);
  235.         write_data('0'+disbuffer[1]);
  236.         write_com(0x82);
  237.         write_data('0'+disbuffer[2]);
  238.         write_com(0x83);
  239.         write_data(disbuffer[3]);
  240.         write_com(0x84);
  241.         write_data('0'+disbuffer[4]);
  242.         write_com(0x85);
  243.         write_data(disbuffer[5]);
  244. }  
  245.   
  246. void diplay_final(void)     //顯示所用到的18b20的實時溫度
  247. {       uint q;   
  248.         for(q=0;q<8;q++)  
  249.             {  
  250.                 RomID_temp[q]=ID[m][q];     //給序列號,等待匹配
  251.             }  
  252.                 Read_Temperature_rom();    //讀取溫度
  253.                 Temperature_cov();         //溫度轉換
  254.                 display();                 //顯示溫度
  255. }
  256. //****************************//
  257. //*****多點18b20搜索子程******//
  258. //****************************//
  259. void search_rom(void)   //遍歷搜索單線上所連的所有18b20的序列號  
  260. {   
  261.   uchar k,l=0,chongtuwei,m,n,a;   
  262.   uchar _00web[MAXNUM]={0};  
  263.   do  
  264.    {  
  265.      DS_init();         //復位單總線上的所有DS18B20  
  266.      write_byte(0xf0);  //單片機發布搜索命令  
  267.      for(m=0;m<8;m++)  
  268.        {  
  269.          uchar s=0; //s用來記錄本次循環得到的1個字節(8位)序列號  
  270.          for(n=0;n<8;n++)  
  271.             {  
  272.              k=read_2bit();  //讀第m*8+n位的原碼和反碼,保存在k中  
  273.              k=k&0x03;       //屏蔽掉k中其它位的干擾,為下一步判斷作準備  
  274.              s>>=1;            //s右移一位,即把上一次循環得到的位值右移一位,  
  275.                              //這樣執行完一次n為變量的循環,便可得到一個字節的ROM號                                                
  276.              if(k==0x01)     //k為01,表明讀到的數據為0,即所有器件在這一位都為0,所以向總線上寫0  
  277.                              //同時對s的值不進行操作,即這位的序列號記為0  
  278.                 {      
  279.                  write_bit (0);  
  280.                 }  
  281.              else if(k==0x02)//k為02,表明讀到的數據為1,即所有器件在這一位都為1,所以向總線上寫1  
  282.                 {  
  283.                  s=s|0x80;  //記錄下此位的值,即s的最高位置1  
  284.                  write_bit (1);  
  285.                 }  
  286.              else if(k==0x00)  
  287.                 {  
  288.                  chongtuwei=m*8+n+1;   //記錄下這個沖突位發生的位置;之所以加1是為了讓_00web數組中的第一位保持0不變,  
  289.                                        //便于判斷搜索循環是否結束;  
  290.                  if(chongtuwei>_00web[l])//如果沖突位比標志00位的位高,即發現了新的沖突位,那么這位寫0  
  291.                     {        
  292.                      write_bit (0);  
  293.                      _00web[++l]=chongtuwei;    //依次記錄位比沖突標志位高的沖突位在_00web數組中   
  294.                     }  
  295.                  else if(chongtuwei<_00web[l]) //如果沖突位比標志00位的位低,那么把ID中這位所在的字節右移n位,  
  296.                     {                        //從而得到這位先前已經寫過的值,如果為0,說明這位先前寫的是0,那么繼續寫0,  
  297.                                             // 如果這位先前寫的是1,那么繼續寫1  
  298.                      a=(ID[num-1][m]>>n)&0x01;  
  299.                      s=s|(a<<7);          //記錄下此位的值  
  300.                      write_bit(a);  
  301.                     }  
  302.                  else if(chongtuwei==_00web[l])//如果沖突位就是標志00位,那么s的最高位置1,即這位記為1,同時向總線上寫1;  
  303.                                              //之所以不寫0,是因為前面已經寫過0,再寫0,就得不到遍歷的效果.  
  304.                     {  
  305.                      s=s|0x80;  
  306.                      write_bit (1);  
  307.                      l=l-1;             //改變標志00位的位置,即向前推一個00位,并且是往低位方向推  
  308.                     }  
  309.                 }  
  310.             }      
  311.          ID[num][m]=s;   
  312.        }  
  313.      num++;         //DS18B20的個數加1  
  314.    }while((_00web[l]!=0)&&(num<MAXNUM));//如果沖突位記錄數組已經前推到0值或是DS18B20的數目已經超過最大允許數目,  
  315.                                       //就退出循環  
  316. }   
  317. //************************//
  318. //*********主程序*********//
  319. //************************//
  320. void main()                 
  321. {     
  322.     delay(10);  
  323.     search_rom();                        
  324.     TMOD = 0x01; //選擇工作方式1
  325.     TH0 = 0x3C;         //設置初始值,定時50MS
  326.     TL0 = 0xB0;
  327.     EA = 1;                         //打開總中斷
  328.     ET0 = 1;                 //打開定時器0中斷
  329.     TR0 = 1;                 //啟動定時器0  
  330.         init();  
  331.     while(1)  
  332.     {     
  333.            display_ROMID();   //顯示序列號
  334.                       display_m();      //顯示18b20個數
  335.            diplay_final();    //顯示溫度
  336.      
  337. …………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼

完整代碼下載:
多點溫度測量.zip (53.69 KB, 下載次數: 76)


評分

參與人數 1黑幣 +3 收起 理由
zzt223 + 3 共享資料的黑幣獎勵!

查看全部評分

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

使用道具 舉報

沙發
ID:435757 發表于 2018-12-4 16:07 | 只看該作者
請問有protues仿真圖么
回復

使用道具 舉報

板凳
ID:366216 發表于 2018-12-6 15:44 | 只看該作者
沒有頭文件代碼嗎,運行不了啊
回復

使用道具 舉報

地板
ID:660749 發表于 2019-12-10 16:36 | 只看該作者
不錯我覺得可以
回復

使用道具 舉報

5#
ID:660749 發表于 2019-12-10 16:37 | 只看該作者
zhangwb055 發表于 2018-12-4 16:07
請問有protues仿真圖么

沒得,自己畫
回復

使用道具 舉報

6#
ID:482246 發表于 2020-2-14 15:10 | 只看該作者
溫度是錯的,老是85
回復

使用道具 舉報

7#
ID:654509 發表于 2020-2-19 15:04 | 只看該作者
不錯,很有幫助
回復

使用道具 舉報

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

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表