制作出來的基于51單片機主控的火災煙霧報警系統實物圖如下:
火災報警 (1).jpg (1.01 MB, 下載次數: 35)
下載附件
2018-7-20 03:33 上傳
1QOK3P%[8M`Q)C5O_{8`U[K.jpg (1.05 MB, 下載次數: 36)
下載附件
2018-7-20 03:33 上傳
G4S[TANIYLA]UYIQ5TX`1$G.jpg (1.15 MB, 下載次數: 28)
下載附件
2018-7-20 03:33 上傳
火災報警 (2).jpg (1.25 MB, 下載次數: 24)
下載附件
2018-7-20 03:33 上傳
火災報警 (3).jpg (1.29 MB, 下載次數: 29)
下載附件
2018-7-20 03:33 上傳
火災報警 USB供電 (1).jpg (872.22 KB, 下載次數: 40)
下載附件
2018-7-20 03:33 上傳
火災報警 USB供電 (2).jpg (1.07 MB, 下載次數: 32)
下載附件
2018-7-20 03:33 上傳
火災報警 散件照片.png (1.35 MB, 下載次數: 25)
下載附件
2018-7-20 03:33 上傳
火災報警.jpg (977.52 KB, 下載次數: 35)
下載附件
2018-7-20 03:33 上傳
單片機火災報警系統電路原理圖如下:
0.png (62.63 KB, 下載次數: 44)
下載附件
2018-7-20 03:34 上傳
元件清單:
0.png (40.17 KB, 下載次數: 35)
下載附件
2018-7-20 03:34 上傳
火災報警系統仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載)
0.png (17.47 KB, 下載次數: 36)
下載附件
2018-7-20 04:50 上傳
單片機源程序如下:
- #include <reg52.h> //調用單片機頭文件
- #define uchar unsigned char //無符號字符型 宏定義 變量范圍0~255
- #define uint unsigned int //無符號整型 宏定義 變量范圍0~65535
- #include <intrins.h>
- #include "eeprom52.h"
- //數碼管段選定義 0 1 2 3 4 5 6 7 8 9
- uchar code smg_du[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,
- 0x88,0x83,0xc6,0xa1,0x86,0x8e,0xff}; //斷碼
- //數碼管位選定義
- uchar code smg_we[]={0x7f,0xbf,0xdf,0xef};
- uchar dis_smg[8] = {0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8};
- sbit CS=P3^2; //CS定義為P3口的第2位腳,連接ADC0832CS腳 PCB
- sbit SCL=P3^3; //SCL定義為P3口的第3位腳,連接ADC0832SCL腳
- sbit DO=P3^4; //DO定義為P3口的第4位腳,連接ADC0832DO腳
- sbit dq = P3^5; //18b20 IO口的定義
- sbit beep = P3^6; //蜂鳴器IO口定義
- uint temperature,s_temp ; //溫度的變量
- uchar dengji,s_dengji; //煙霧等級
- uchar shoudong; //手動報警鍵
- bit flag_300ms = 1;
- uchar key_can; //按鍵值的變量
- uchar menu_1; //菜單設計的變量
- /***********************1ms延時函數*****************************/
- void delay_1ms(uint q)
- {
- uint i,j;
- for(i=0;i<q;i++)
- for(j=0;j<120;j++);
- }
- /***********************小延時函數*****************************/
- void delay_uint(uint q)
- {
- while(q--);
- }
- /******************把數據保存到單片機內部eeprom中******************/
- void write_eeprom()
- {
- SectorErase(0x2000);
- byte_write(0x2000, s_temp);
- byte_write(0x2001, s_dengji);
- byte_write(0x2060, a_a);
- }
- /******************把數據從單片機內部eeprom中讀出來*****************/
- void read_eeprom()
- {
- s_temp = byte_read(0x2000);
- s_dengji = byte_read(0x2001);
- a_a = byte_read(0x2060);
- }
- /**************開機自檢eeprom初始化*****************/
- void init_eeprom()
- {
- read_eeprom(); //先讀
- if(a_a != 1) //新的單片機初始單片機內問eeprom
- {
- s_temp = 50;
- s_dengji = 5;
- a_a = 1;
- write_eeprom(); //保存數據
- }
- }
- /***********************18b20初始化函數*****************************/
- void init_18b20()
- {
- bit q;
- dq = 1; //把總線拿高
- delay_uint(1); //15us
- dq = 0; //給復位脈沖
- delay_uint(80); //750us
- dq = 1; //把總線拿高 等待
- delay_uint(10); //110us
- q = dq; //讀取18b20初始化信號
- delay_uint(20); //200us
- dq = 1; //把總線拿高 釋放總線
- }
- /*************寫18b20內的數據***************/
- void write_18b20(uchar dat)
- {
- uchar i;
- for(i=0;i<8;i++)
- { //寫數據是低位開始
- dq = 0; //把總線拿低寫時間隙開始
- dq = dat & 0x01; //向18b20總線寫數據了
- delay_uint(5); // 60us
- dq = 1; //釋放總線
- dat >>= 1;
- }
- }
- /*************讀取18b20內的數據***************/
- uchar read_18b20()
- {
- uchar i,value;
- for(i=0;i<8;i++)
- {
- dq = 0; //把總線拿低讀時間隙開始
- value >>= 1; //讀數據是低位開始
- dq = 1; //釋放總線
- if(dq == 1) //開始讀寫數據
- value |= 0x80;
- delay_uint(5); //60us 讀一個時間隙最少要保持60us的時間
- }
- return value; //返回數據
- }
- /*************讀取溫度的值 讀出來的是小數***************/
- uint read_temp()
- {
- uint value;
- uchar low; //在讀取溫度的時候如果中斷的太頻繁了,就應該把中斷給關了,否則會影響到18b20的時序
- init_18b20(); //初始化18b20
- write_18b20(0xcc); //跳過64位ROM
- write_18b20(0x44); //啟動一次溫度轉換命令
- delay_uint(50); //500us
- init_18b20(); //初始化18b20
-
- write_18b20(0xcc); //跳過64位ROM
- write_18b20(0xbe); //發出讀取暫存器命令
-
- EA = 0;
- low = read_18b20(); //讀溫度低字節
- value = read_18b20(); //讀溫度高字節
- EA = 1;
- value <<= 8; //把溫度的高位左移8位
- value |= low; //把讀出的溫度低位放到value的低八位中
- value *= 0.0625; //轉換到溫度值
- return value; //返回讀出的溫度
- }
- /***********讀數模轉換數據********************************************************/
- //請先了解ADC0832模數轉換的串行協議,再來讀本函數,主要是對應時序圖來理解,本函數是模擬0832的串行協議進行的
- unsigned char ad0832read(bit SGL,bit ODD)
- {
- unsigned char i=0,value=0,value1=0;
- SCL=0;
- DO=1;
- CS=0; //開始
- SCL=1; //第一個上升沿
- SCL=0;
- DO=SGL;
- SCL=1; //第二個上升沿
- SCL=0;
- DO=ODD;
- SCL=1; //第三個上升沿
- SCL=0; //第三個下降沿
- DO=1;
- for(i=0;i<8;i++)
- {
- SCL=1;
- SCL=0; //開始從第四個下降沿接收數據
- value<<=1;
- if(DO)
- value++;
- }
- for(i=0;i<8;i++)
- { //接收校驗數據
- value1>>=1;
- if(DO)
- value1+=0x80;
- SCL=1;
- SCL=0;
- }
- CS=1;
- SCL=1;
- if(value==value1) //與校驗數據比較,正確就返回數據,否則返回0
- return value;
- return 0;
- }
- /***********************數碼顯示函數*****************************/
- void display()
- {
- uchar i;
- P1 = 0xff; //消隱
- P2 = smg_we[i]; //位選
- P1 = dis_smg[i]; //段選
- i ++;
- if(i >= 4) //4位數碼管顯示
- i = 0;
- }
- /*************定時器0初始化程序***************/
- void time_init()
- {
- EA = 1; //開總中斷
- TMOD = 0X01; //定時器0、定時器1工作方式1
- ET0 = 1; //開定時器0中斷
- TR0 = 1; //允許定時器0定時
- }
- /********************獨立按鍵程序*****************/
- uchar key_can; //按鍵值
- void key() //獨立按鍵程序
- {
- static uchar key_new;
- key_can = 20; //按鍵值還原
- P2 |= 0x0f;
- if((P2 & 0x0f) != 0x0f) //按鍵按下
- {
- delay_1ms(1); //按鍵消抖動
- if(((P2 & 0x0f) != 0x0f) && (key_new == 1))
- { //確認是按鍵按下
- key_new = 0;
- switch(P2 & 0x0f)
- {
- case 0x0e: key_can = 4; break; //得到k1鍵值
- case 0x0d: key_can = 3; break; //得到k2鍵值
- case 0x0b: key_can = 2; break; //得到k3鍵值
- case 0x07: key_can = 1; break; //得到k4鍵值
- }
- }
- }
- else //按鍵松開
- key_new = 1;
- }
- /****************按鍵處理數碼管顯示函數***************/
- void key_with()
- {
- if(key_can == 4) //緊急報警鍵 手動報警
- {
- if(menu_1 == 0)
- shoudong = 1;
- }
- if(key_can == 1) //設置鍵
- {
- menu_1 ++;
- if(menu_1 >= 3)
- {
- menu_1 = 0;
- }
- }
- if(menu_1 == 0)
- {
- if((key_can == 2) || (key_can == 3))
- shoudong = 0; //取消手動報警
- }
- if(menu_1 == 1) //設置高溫報警
- {
- if(key_can == 2)
- {
- s_temp ++ ; //高溫報警值加1
- if(s_temp > 99)
- s_temp = 99;
- }
- if(key_can == 3)
- {
- s_temp -- ; //高溫報警值減1
- if(s_temp <= 10)
- s_temp = 10 ;
- }
- dis_smg[0] = smg_du[s_temp % 10]; //取個位顯示
- dis_smg[1] = smg_du[s_temp / 10 % 10]; //取十位顯示
- dis_smg[2] = 0xbf;
- dis_smg[3] = smg_du[10]; //顯示A
- write_eeprom(); //保存數據
- }
- if(menu_1 == 2) //設置煙霧報警
- {
- if(key_can == 2)
- {
- s_dengji ++ ; //煙霧報警值加1
- if(s_dengji >= 9)
- s_dengji = 9;
- }
- if(key_can == 3)
- {
- s_dengji --; //煙霧報警值減1
- if(s_dengji <= 1)
- s_dengji = 1;
- }
- dis_smg[0] = smg_du[s_dengji % 10]; //取個位顯示
- dis_smg[1] = 0xbf ;
- dis_smg[2] = 0xbf;
- dis_smg[3] = smg_du[11]; //顯示B
- write_eeprom(); //保存數據
- }
- }
- /****************報警函數***************/
- void clock_h_l()
- {
- static uchar value;
- if((dengji >= s_dengji) || (temperature >= s_temp) || (shoudong == 1)) //報警
- {
- value ++;
- if(value >= 2)
- {
- value = 10;
- beep = ~beep; //蜂鳴器報警
- }
- }
- else
- {
- if((dengji < s_dengji) && (temperature < s_temp) && (shoudong == 0)) //取消報警
- {
- value = 0;
- beep = 1; //取消報警
- }
- }
- }
- /***************主函數*****************/
- void main()
- {
- beep = 0; //開機蜂鳴器叫一聲
- delay_1ms(200);
- P0 = P1 = P2 = P3 = 0xff; //初始化IO口為高電平
- temperature = read_temp(); //讀取溫度值
- init_eeprom(); //開始初始化保存的數據
- delay_1ms(650);
- temperature = read_temp(); //讀取溫度值
- time_init(); //初始化定時器
- while(1)
- {
- ……………………
- …………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼
設計文檔:
0.png (31.32 KB, 下載次數: 43)
下載附件
2018-7-20 04:55 上傳
所有資料51hei提供下載:
資料 火災報警 聲光報警.7z
(6.26 MB, 下載次數: 332)
2023-4-3 16:35 上傳
點擊文件名下載附件
火災報警 下載積分: 黑幣 -5
|