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

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

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

單片機傳感器設計 火災報警器 Proteus仿真 含程序.hex文件,仿真圖

  [復制鏈接]
跳轉到指定樓層
樓主
ID:786476 發表于 2020-6-22 11:22 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
完整文檔下載(word格式 可編輯):仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載)


單片機源程序如下:
  1. #include <reg52.h>
  2. #include "intrins.h"                                                                                          

  3. #define     u8                          unsigned char
  4. #define     u16                   unsigned int
  5. #define     uchar                          unsigned char
  6. #define     uint                   unsigned int

  7. uchar yushe_wendu=50;                                //溫度預設值
  8. uchar yushe_yanwu=100;                                //煙霧預設值
  9. uint wendu;                                                                 //溫度值全局變量
  10. uchar yanwu;                                                                 //用于讀取ADC數據

  11. //運行模式  
  12. uchar Mode=0;                                                                 //=1是設置溫度閥值  =2是設置煙霧閥值        =0是正常監控模式
  13. //管腳聲明
  14. sbit Led_Reg                 =P2^2;                                 //紅燈
  15. sbit Led_Yellow  =P2^4;                                 //黃燈
  16. sbit Buzzer             =P2^0;                                 //蜂鳴器
  17. sbit Fan                      =P3^3;                                 //


  18. /********************************************************************
  19. * 名稱 : delay_1ms()
  20. * 功能 : 延時1ms函數
  21. * 輸入 : q
  22. * 輸出 : 無
  23. ***********************************************************************/
  24. void delay_ms(uint q)
  25. {
  26.         uint i,j;
  27.         for(i=0;i<q;i++)
  28.                 for(j=0;j<110;j++);
  29. }
  30. /***********************************************************************************************************
  31. LCD1602相關函數
  32. ***********************************************************************************************************/

  33. //LCD管腳聲明
  34. sbit LCDRS = P2^7;
  35. sbit LCDEN = P2^6;
  36. sbit D0                 = P0^0;
  37. sbit D1                 = P0^1;
  38. sbit D2                 = P0^2;
  39. sbit D3                 = P0^3;
  40. sbit D4                 = P0^4;
  41. sbit D5                 = P0^5;
  42. sbit D6                 = P0^6;
  43. sbit D7                 = P0^7;



  44. //LCD延時
  45. void LCDdelay(uint z)                  //該延時大約100us
  46. {
  47.   uint x,y;
  48.   for(x=z;x>0;x--)
  49.     for(y=10;y>0;y--);
  50. }
  51. void LCD_WriteData(u8 dat)          
  52. {
  53.         if(dat&0x01)D0=1;else D0=0;
  54.         if(dat&0x02)D1=1;else D1=0;
  55.         if(dat&0x04)D2=1;else D2=0;
  56.         if(dat&0x08)D3=1;else D3=0;
  57.         if(dat&0x10)D4=1;else D4=0;
  58.         if(dat&0x20)D5=1;else D5=0;
  59.         if(dat&0x40)D6=1;else D6=0;
  60.         if(dat&0x80)D7=1;else D7=0;
  61. }
  62. //命令
  63. void write_com(uchar com)
  64. {
  65.   LCDRS=0;                                  
  66.         LCD_WriteData(com);
  67. //  DAT=com;
  68.   LCDdelay(5);
  69.   LCDEN=1;
  70.   LCDdelay(5);
  71.   LCDEN=0;
  72. }
  73. //數據
  74. void write_data(uchar date)
  75. {
  76.   LCDRS=1;
  77.         LCD_WriteData(date);
  78. //  DAT=date;
  79.   LCDdelay(5);
  80.   LCDEN=1;
  81.   LCDdelay(5);
  82.   LCDEN=0;
  83. }

  84. /*------------------------------------------------
  85.               選擇寫入位置
  86. ------------------------------------------------*/
  87. void SelectPosition(unsigned char x,unsigned char y)
  88. {     
  89.         if (x == 0)
  90.         {     
  91.                 write_com(0x80 + y);     //表示第一行
  92.         }
  93.         else
  94.         {      
  95.                 write_com(0xC0 + y);      //表示第二行
  96.         }        
  97. }
  98. /*------------------------------------------------
  99.               寫入字符串函數
  100. ------------------------------------------------*/
  101. void LCD_Write_String(unsigned char x,unsigned char y,unsigned char *s)
  102. {     
  103.         SelectPosition(x,y) ;
  104.         while (*s)
  105.         {     
  106.                 write_data( *s);     
  107.                 s ++;     
  108.         }
  109. }
  110. //========================================================================
  111. // 函數: void LCD_Write_Char(u8 x,u8 y,u16 s,u8 l)
  112. // 應用: LCD_Write_Char(0,1,366,4) ;
  113. // 描述: 在第0行第一個字節位置顯示366的后4位,顯示結果為 0366
  114. // 參數: x:行,y:列,s:要顯示的字,l:顯示的位數
  115. // 返回: none.
  116. // 版本: VER1.0
  117. // 日期: 2013-4-1
  118. // 備注: 最大顯示65535
  119. //========================================================================
  120. void LCD_Write_Char(u8 x,u8 y,u16 s,u8 l)
  121. {     
  122.         SelectPosition(x,y) ;

  123.         if(l>=5)
  124.                 write_data(0x30+s/10000%10);        //萬位
  125.         if(l>=4)
  126.                 write_data(0x30+s/1000%10);                //千位
  127.         if(l>=3)
  128.                 write_data(0x30+s/100%10);                //百位
  129.         if(l>=2)
  130.                 write_data(0x30+s/10%10);                        //十位
  131.         if(l>=1)
  132.                 write_data(0x30+s%10);                //個位

  133. }
  134. /*1602指令簡介
  135.   write_com(0x38);//屏幕初始化
  136.   write_com(0x0c);//打開顯示 無光標 無光標閃爍
  137.   write_com(0x0d);//打開顯示 陰影閃爍
  138.   write_com(0x0d);//打開顯示 陰影閃爍
  139. */
  140. //1602初始化
  141. void Init1602()
  142. {
  143.   uchar i=0;
  144.   write_com(0x38);//屏幕初始化
  145.   write_com(0x0c);//打開顯示 無光標 無光標閃爍
  146.   write_com(0x06);//當讀或寫一個字符是指針后一一位
  147.   write_com(0x01);//清屏
  148.        
  149. }

  150. void Display_1602(yushe_wendu,yushe_yanwu,c,temp)
  151. {
  152.         //顯示預設溫度
  153.         LCD_Write_Char(0,6,yushe_wendu,2) ;
  154.        
  155.         //顯示預設煙霧
  156.         LCD_Write_Char(0,13,yushe_yanwu,3) ;
  157.        
  158.         //時時溫度
  159.         LCD_Write_Char(1,6,c/10,2) ;
  160.         write_data('.');
  161.         LCD_Write_Char(1,9,c%10,1) ;
  162.        
  163.         //時時煙霧
  164.         LCD_Write_Char(1,13,temp,3) ;
  165. }



  166. /***********************************************************************************************************
  167. ADC0832相關函數
  168. ***********************************************************************************************************/
  169. sbit ADCS         =P1^5; //ADC0832 片選
  170. sbit ADCLK  =P1^2; //ADC0832 時鐘
  171. sbit ADDI         =P1^3; //ADC0832 數據輸入                /*因為單片機的管腳是雙向的,且ADC0832的數據輸入輸出不同時進行,
  172. sbit ADDO         =P1^3; //ADC0832 數據輸出                /*為節省單片機引腳,簡化電路所以輸入輸出連接在同一個引腳上



  173. //========================================================================
  174. // 函數: unsigned int Adc0832(unsigned char channel)
  175. // 應用:                 temp=Adc0832(0);
  176. // 描述: 讀取0通道的AD值
  177. // 參數: channel:通道0和通道1選擇
  178. // 返回: 選取通道的AD值
  179. // 版本: VER1.0
  180. // 日期: 2015-05-29
  181. // 備注:
  182. //========================================================================
  183. unsigned int Adc0832(unsigned char channel)
  184. {
  185.         uchar i=0;
  186.         uchar j;
  187.         uint dat=0;
  188.         uchar ndat=0;
  189.         uchar  Vot=0;

  190.         if(channel==0)channel=2;
  191.         if(channel==1)channel=3;
  192.         ADDI=1;
  193.         _nop_();
  194.         _nop_();
  195.         ADCS=0;//拉低CS端
  196.         _nop_();
  197.         _nop_();
  198.         ADCLK=1;//拉高CLK端
  199.         _nop_();
  200.         _nop_();
  201.         ADCLK=0;//拉低CLK端,形成下降沿1
  202.         _nop_();
  203.         _nop_();
  204.         ADCLK=1;//拉高CLK端
  205.         ADDI=channel&0x1;
  206.         _nop_();
  207.         _nop_();
  208.         ADCLK=0;//拉低CLK端,形成下降沿2
  209.         _nop_();
  210.         _nop_();
  211.         ADCLK=1;//拉高CLK端
  212.         ADDI=(channel>>1)&0x1;
  213.         _nop_();
  214.         _nop_();
  215.         ADCLK=0;//拉低CLK端,形成下降沿3
  216.         ADDI=1;//控制命令結束
  217.         _nop_();
  218.         _nop_();
  219.         dat=0;
  220.         for(i=0;i<8;i++)
  221.         {
  222.                 dat|=ADDO;//收數據
  223.                 ADCLK=1;
  224.                 _nop_();
  225.                 _nop_();
  226.                 ADCLK=0;//形成一次時鐘脈沖
  227.                 _nop_();
  228.                 _nop_();
  229.                 dat<<=1;
  230.                 if(i==7)dat|=ADDO;
  231.         }
  232.         for(i=0;i<8;i++)
  233.         {
  234.                 j=0;
  235.                 j=j|ADDO;//收數據
  236.                 ADCLK=1;
  237.                 _nop_();
  238.                 _nop_();
  239.                 ADCLK=0;//形成一次時鐘脈沖
  240.                 _nop_();
  241.                 _nop_();
  242.                 j=j<<7;
  243.                 ndat=ndat|j;
  244.                 if(i<7)ndat>>=1;
  245.         }
  246.         ADCS=1;//拉低CS端
  247.         ADCLK=0;//拉低CLK端
  248.         ADDO=1;//拉高數據端,回到初始狀態
  249.         dat<<=8;
  250.         dat|=ndat;

  251.         return(dat);            //return ad data
  252. }

  253. /***********************************************************************************************************
  254. DS18B20相關函數
  255. ***********************************************************************************************************/

  256. sbit DQ = P1^0;                                 //ds18b20的數據引腳



  257. /*****延時子程序:該延時主要用于ds18b20延時*****/
  258. void Delay_DS18B20(int num)
  259. {
  260.   while(num--) ;
  261. }
  262. /*****初始化DS18B20*****/
  263. void Init_DS18B20(void)
  264. {
  265.   unsigned char x=0;
  266.   DQ = 1;         //DQ復位
  267.   Delay_DS18B20(8);    //稍做延時
  268.   DQ = 0;         //單片機將DQ拉低
  269.   Delay_DS18B20(80);   //精確延時,大于480us
  270.   DQ = 1;         //拉高總線
  271.   Delay_DS18B20(14);
  272.   x = DQ;           //稍做延時后,如果x=0則初始化成功,x=1則初始化失敗
  273.   Delay_DS18B20(20);
  274. }
  275. /*****讀一個字節*****/
  276. unsigned char ReadOneChar(void)
  277. {
  278.   unsigned char i=0;
  279.   unsigned char dat = 0;
  280.   for (i=8;i>0;i--)
  281.   {
  282.     DQ = 0;     // 給脈沖信號
  283.     dat>>=1;
  284.     DQ = 1;     // 給脈沖信號
  285.     if(DQ)
  286.     dat|=0x80;
  287.     Delay_DS18B20(4);
  288.   }
  289.   return(dat);
  290. }
  291. /*****寫一個字節*****/
  292. void WriteOneChar(unsigned char dat)
  293. {
  294.   unsigned char i=0;
  295.   for (i=8; i>0; i--)
  296.   {
  297.     DQ = 0;
  298.     DQ = dat&0x01;
  299.     Delay_DS18B20(5);
  300.     DQ = 1;
  301.     dat>>=1;
  302.   }
  303. }
  304. /*****讀取溫度*****/
  305. unsigned int ReadTemperature(void)
  306. {
  307.   unsigned char a=0;
  308.   unsigned char b=0;
  309.   unsigned int t=0;
  310.   float tt=0;
  311.   Init_DS18B20();
  312.   WriteOneChar(0xCC);  //跳過讀序號列號的操作
  313.   WriteOneChar(0x44);  //啟動溫度轉換
  314.   Init_DS18B20();
  315.   WriteOneChar(0xCC);  //跳過讀序號列號的操作
  316.   WriteOneChar(0xBE);  //讀取溫度寄存器
  317.   a=ReadOneChar();     //讀低8位
  318.   b=ReadOneChar();    //讀高8位
  319.   t=b;
  320.   t<<=8;
  321.   t=t|a;
  322.   tt=t*0.0625;
  323.   t= tt*10+0.5;     //放大10倍輸出并四舍五入
  324.   return(t);
  325. }
  326. //=====================================================================================
  327. //=====================================================================================
  328. //=====================================================================================


  329. /*****校準溫度*****/
  330. u16 check_wendu(void)
  331. {
  332.         u16 c;
  333.         c=ReadTemperature()-5;                          //獲取溫度值并減去DS18B20的溫漂誤差
  334.         if(c<1) c=0;
  335.         if(c>=999) c=999;
  336.         return c;
  337. }


  338. /***********************************************************************************************************
  339. 按鍵檢測相關函數
  340. ***********************************************************************************************************/
  341. //按鍵
  342. sbit Key1=P1^6;                               
  343. sbit Key2=P1^7;                               
  344. sbit Key3=P3^2;                               


  345. #define KEY_SET                 1               
  346. #define KEY_ADD                        2               
  347. #define KEY_MINUS                3               

  348. //========================================================================
  349. // 函數: u8 Key_Scan()
  350. // 應用: temp=u8 Key_Scan();
  351. // 描述: 按鍵掃描并返回按下的鍵值
  352. // 參數: NONE
  353. // 返回: 按下的鍵值
  354. // 版本: VER1.0
  355. // 日期: 2015-05-29
  356. // 備注: 該函數帶松手檢測,按下鍵返回一次鍵值后返回0,直至第二次按鍵按下
  357. //========================================================================
  358. u8 Key_Scan()
  359. {         
  360.         static u8 key_up=1;//按鍵按松開標志
  361.         if(key_up&&(Key1==0||Key2==0||Key3==0))
  362.         {
  363.                 delay_ms(10);//去抖動
  364.                 key_up=0;
  365.                 if(Key1==0)                        return 1;
  366.                 else if(Key2==0)return 2;
  367.                 else if(Key3==0)return 3;
  368.         }
  369.         else if(Key1==1&&Key2==1&&Key3==1)
  370.                 key_up=1;             
  371.         return 0;// 無按鍵按下
  372. }



  373. void main (void)
  374. {
  375.         u8 key;
  376.         wendu=check_wendu();                  //初始化時調用溫度讀取函數 防止開機85°C
  377.         Init1602();                          //調用初始化顯示函數
  378.         LCD_Write_String(0,0,"SET T:00   E:000");  //開機界面
  379.         LCD_Write_String(1,0,"NOW T:00.0 E:000");  
  380.         delay_ms(1000);
  381.         wendu=check_wendu();                  //初始化時調用溫度讀取函數 防止開機85°C
  382.         while (1)                                                //主循環
  383.         {
  384.                 key=Key_Scan();                                        //按鍵掃描
  385.                 yanwu=Adc0832(0);                                //讀取煙霧值
  386.                 wendu=check_wendu();          //讀取溫度值
  387.                
  388.                 if(key==KEY_SET)
  389.                 {
  390.                         Mode++;
  391.                 }
  392.                
  393.                 switch(Mode)                                                //判斷模式的值
  394.                 {
  395.                         case 0:                                                                //監控模式
  396.                         {
  397.                                 Display_1602(yushe_wendu,yushe_yanwu,wendu,yanwu);  //顯示預設溫度,預設煙霧,溫度值,煙霧值
  398.                                 if(yanwu>=yushe_yanwu)          //煙霧值大于等于預設值時
  399.                                 {
  400.                                         Led_Reg=0;                                          //煙霧指示燈亮
  401.                                         Fan=0;
  402.                                         Buzzer=0;                                                  //蜂鳴器報警
  403.                                 }
  404.                                 else                                                                                  //煙霧值小于預設值時
  405.                                 {
  406.                                         Led_Reg=1;                                          //關掉報警燈
  407.                                         Fan=1;
  408.                                 }
  409.                                 if(wendu>=(yushe_wendu*10))          //溫度大于等于預設溫度值時(為什么是大于預設值*10:因為我們要顯示的溫度是有小數點后一位,是一個3位數,25.9°C時實際讀的數是259,所以判斷預設值時將預設值*10)
  410.                                 {
  411.                                         Buzzer=0;                                                  //打開蜂鳴器報警
  412.                                         Led_Yellow=0;                                          //打開溫度報警燈
  413.                                 }
  414.                                 else                                                                                  //溫度值小于預設值時
  415.                                 {
  416.                                         Led_Yellow=1;                                          //關閉報警燈
  417.                                 }
  418.                                 if((yanwu<yushe_yanwu)&&(wendu<(yushe_wendu*10)))          //當煙霧小于預設值并且溫度也小于預設值時 (&&:邏輯與,左右兩邊的表達式都成立(都為真,也就是1)時,該if語句才成立)
  419.                                 {
  420.                                         Buzzer=1;                                                  //停止報警
  421.                                 }
  422.                                 break;
  423.                         }
  424.                         case 1://預設溫度模式
  425.                         {
  426.                                 SelectPosition(0,5) ;                                        //指定位置
  427.                            write_com(0x0d);                                                        //陰影閃爍
  428.                                 if(key==KEY_ADD)                                                        //加鍵按下
  429.                                 {
  430.                                         yushe_wendu++;                                            //預設溫度值(閥值)加1
  431.                                         if(yushe_wendu>=99)                                         //當閥值加到大于等于99時
  432.                                         yushe_wendu=99;                                                         //閥值固定為99
  433.                                         LCD_Write_Char(0,6,yushe_wendu,2) ;//顯示預設溫度
  434.                                 }
  435.                                 if(key==KEY_MINUS)                                                 //減鍵按下
  436.                                 {
  437.                                         if(yushe_wendu<=1)                                        //當溫度上限值減小到1時
  438.                                         yushe_wendu=1;                          //固定為1
  439.                                         yushe_wendu--;                                                        //預設溫度值減一,最小為0
  440.                                         LCD_Write_Char(0,6,yushe_wendu,2) ;//顯示預設溫度
  441.                                 }
  442.                                 break;                                                                                          //執行后跳出switch
  443.                         }
  444.                         case 2:                                //預設煙霧模式
  445.                         {
  446.                                 SelectPosition(0,12) ;                                //指定位置       
  447.                            write_com(0x0d);                                                        //打開顯示 無光標 光標閃爍
  448.                                 if(key==KEY_ADD)                                                        //加鍵按下
  449.                                 {
  450.                                         if(yushe_yanwu>=255)        //當閥值加到大于等于255時
  451.                                         yushe_yanwu=254;            //閥值固定為254
  452.                                         yushe_yanwu++;                                            //預設煙霧值(閥值)加1,最大為255
  453.                                         LCD_Write_Char(0,13,yushe_yanwu,3) ;//顯示預設煙霧
  454.                                 }
  455.                                 if(key==KEY_MINUS)                                                //減鍵按下
  456.                                 {
  457.                                         if(yushe_yanwu<=1)                                        //當煙霧上限值減小到1時
  458.                                                 yushe_yanwu=1;                  //固定為1
  459.                                         yushe_yanwu--;                                                        //預設溫度值減一,最小為0          
  460.                                         LCD_Write_Char(0,13,yushe_yanwu,3) ;//顯示預設煙霧
  461.                                 }
  462.                                 break;
  463.                         }
  464.                         default        :       
  465.                         {
  466.                                 write_com(0x38);//屏幕初始化
  467.                                 write_com(0x0c);//打開顯示 無光標 無光標閃爍
  468.                                 Mode=0;                        //恢復正常模式
  469.                                 break;
  470.                         }
  471.                 }
  472.                
  473.         }

  474. }








復制代碼

所有資料51hei提供下載:
新建壓縮文件.zip (31.46 KB, 下載次數: 241)
火災報警器.doc (178.04 KB, 下載次數: 139)



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

使用道具 舉報

來自 2#
ID:262 發表于 2021-12-3 18:41 | 只看該作者
q2789456123 發表于 2021-12-3 08:40
程序里面nop未定義

估計是你缺少頭文件導致的,建議重新安裝一下Keil5即可,你看我編譯的樓主的程序,我上傳了我的工程文件,你去下載研究下吧


51hei.gif (95.37 KB, 下載次數: 117)

51hei.gif

c樣本51hei.7z

58.81 KB, 下載次數: 44, 下載積分: 黑幣 -5

回復

使用道具 舉報

板凳
ID:888732 發表于 2021-3-22 13:12 | 只看該作者
很好 幫助非常大
回復

使用道具 舉報

地板
ID:987283 發表于 2021-12-3 08:40 | 只看該作者
程序里面nop未定義
回復

使用道具 舉報

5#
ID:988274 發表于 2021-12-6 21:28 | 只看該作者
顯示屏不顯示數是為什么啊
回復

使用道具 舉報

6#
ID:262 發表于 2021-12-11 03:03 | 只看該作者
1064664752 發表于 2021-12-6 21:28
顯示屏不顯示數是為什么啊

去下載我編譯好的程序 Proteus用8.8版本
回復

使用道具 舉報

7#
ID:992548 發表于 2021-12-15 16:25 | 只看該作者
我用不了 顯示屏NOW那欄顯示: NOW T:99.9怎么辦啊
回復

使用道具 舉報

8#
ID:1036068 發表于 2022-6-21 10:18 來自手機 | 只看該作者
51單片機霍爾測速與PWM調直流電機轉速快慢
回復

使用道具 舉報

9#
ID:63541 發表于 2022-10-4 17:34 | 只看該作者
我的Proteus 8運行完美.謝謝
回復

使用道具 舉報

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

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 国产黄色网址在线观看 | 一区二区三区在线 | 中文字幕亚洲一区 | av日韩高清 | 亚洲精品久久视频 | 天天躁人人躁人人躁狂躁 | 黄网站免费观看 | 日本视频免费观看 | 91高清视频在线观看 | 精品国产青草久久久久96 | 午夜男人天堂 | 国产丝袜一区二区三区免费视频 | 日韩精品视频一区二区三区 | 一区二区三区四区在线免费观看 | 日本黄色一级片视频 | 九九热这里只有精品在线观看 | 欧美日韩在线免费观看 | 久久久精品 | 一级黄色短片 | 看a网站 | 国产精品一区二区三 | 中文字幕在线视频一区二区三区 | 在线观看成年人视频 | 欧美一区二区三区免费在线观看 | 欧美久久影院 | 日日操操 | 欧美精品一区三区 | 国产在线一区二区 | 久久久久久久一区二区 | 国产精品免费在线 | 超碰97人人人人人蜜桃 | 国产精品欧美一区二区三区不卡 | 国产精品自拍视频网站 | 最新中文字幕在线 | 日韩色视频| 人人澡人人射 | 免费视频成人国产精品网站 | 91精品国产乱码久久久 | 一级毛片播放 | 亚洲国产乱码 | 一区二区三区不卡视频 |