基于51单片机的温度烟雾火灾报警系统
(仿真+程序+PCB+设计报告)
功能介绍
具体功能:
1. 烟雾采集传感器选用MQ-2,由于该传感器输出的是模拟信号,因此需要模数转换芯片ADC0832,把模拟信号转为数字信号再传给单片机;
2. 温度采集使用DS18B20数字传感器,该传感器通过单总线和单片机相连;
3. 显示设备使用数码管,可以在屏幕上显示实时烟雾浓度等级、温度值;
4. 报警模块则采用蜂鸣器和LED灯组成,无论哪个检测元素超标,蜂鸣器都会鸣叫报警;
5. 3个按键作为操作输入设备,可以用来调节报警阈值的大小;
6. 增加了一个手动报警按键功能。
演示视频:
基于51单片机的温度烟雾火灾报警系统
#include <reg52.h> //调用单片机头文件
#define uchar unsigned char //无符号字符型 宏定义 变量范围0~255
#define uint unsigned int //无符号整型 宏定义 变量范围0~65535
#include <intrins.h>
//数码管段选定义 0 1 2 3 4 5 6 7 8 9
uchar code smg_du[]={0x05,0xdd,0x46,0x54,0x9c,0x34,0x24,0x5d,0x04,0x14,
0x0c,0xa4,0x27,0xc4,0x26,0x2e,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定义为P1口的第4位脚,连接ADC0832CS脚 PCB
sbit SCL=P3^3; //SCL定义为P1口的第3位脚,连接ADC0832SCL脚
sbit DO=P3^4; //DO定义为P1口的第4位脚,连接ADC0832DO脚
sbit dq = P3^5; //18b20 IO口的定义
sbit beep = P3^6; //蜂鸣器IO口定义
uint temperature,s_temp=35; //温度的变量
uchar dengji,s_dengji=5; //烟物等级
uchar shoudong; //手动报警键
uchar flag_huoyan;
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--);
}
/***********************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(unsigned char channel)
{
uchar i=0;
uchar j;
uint dat=0;
uchar ndat=0;
uchar Vot=0;
if(channel==0)channel=2;
if(channel==1)channel=3;
DO=1;
_nop_();
_nop_();
CS=0;//拉低CS端
_nop_();
_nop_();
SCL=1;//拉高CLK端
_nop_();
_nop_();
SCL=0;//拉低CLK端,形成下降沿1
_nop_();
_nop_();
SCL=1;//拉高CLK端
DO=channel&0x1;
_nop_();
_nop_();
SCL=0;//拉低CLK端,形成下降沿2
_nop_();
_nop_();
SCL=1;//拉高CLK端
DO=(channel>>1)&0x1;
_nop_();
_nop_();
SCL=0;//拉低CLK端,形成下降沿3
DO=1;//控制命令结束
_nop_();
_nop_();
dat=0;
for(i=0;i<8;i++)
{
dat|=DO;//收数据
SCL=1;
_nop_();
_nop_();
SCL=0;//形成一次时钟脉冲
_nop_();
_nop_();
dat<<=1;
if(i==7)dat|=DO;
}
for(i=0;i<8;i++)
{
j=0;
j=j|DO;//收数据
SCL=1;
_nop_();
_nop_();
SCL=0;//形成一次时钟脉冲
_nop_();
_nop_();
j=j<<7;
ndat=ndat|j;
if(i<7)ndat>>=1;
}
CS=1;//拉低CS端
SCL=0;//拉低CLK端
DO=1;//拉高数据端,回到初始状态
dat<<=8;
dat|=ndat;
return(dat); //return ad data
}
/***********************数码显示函数*****************************/
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] = 0xfe;
dis_smg[3] = smg_du[10]; //显示A
}
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] = 0xfe ;
dis_smg[2] = 0xfe;
dis_smg[3] = smg_du[11]; //显示B
}
}
/****************报警函数***************/
void clock_h_l()
{
static uchar value;
if((dengji >= s_dengji) || (temperature >= s_temp) || (shoudong == 1) || (flag_huoyan == 1)) //报警
{
value ++;
if(value >= 2)
{
value = 10;
beep = ~beep; //蜂鸣器报警
}
}else
{
if((dengji < s_dengji) && (temperature < s_temp) && (shoudong == 0) && (flag_huoyan == 0)) //取消报警
{
value = 0;
beep = 1; //取消报警
}
}
}
硬件设计
使用元器件:
单片机:STC89C52;
(注意:单片机是通用的,无论51还是52、无论stc还是at都一样,引脚功能都一样。程序也是一样的。)
蜂鸣器;万能板;共阳极数码管;
PNP三极管;发光二极管;
ds18b20; 晶振:11.0592;
电容 :20pF,1uF;
电阻:51Ω,4.7k;
按键开关;ADC0832;
导线:若干;
流程图:
设计资料
01 仿真图
本设计使用proteus8.7版本设计,具体如图!
02 原理图
本系统原理图采用Altium Designer19设计,具体如图!
03 程序
本设计使用软件keil4和keil5两个版本编程设计,无需担心!具体如图!
04 设计报告
一万两千字设计报告,具体如下!
05 设计资料
资料获取请关注同名公众号,全部资料包括仿真源文件 、程序(含注释)、AD原理图、参考论文、流程图等。具体内容如下,全网最全! !
资料获取请观看前面演示视频!
点赞分享一起学习成长。