基于51单片机的家居空气质量监控系统的仿真原理图程序设计

系统硬件设计

(末尾附文件)
整个系统以STC89C52单片机为核心器件,配合电阻电容晶振等器件,构成单片机的最小系统。其它个模块围绕着单片机最小系统展开。其中包括,温湿度传感器输采用DHT11温湿度一体的传感器,负责采集温度和湿度的数据后发给单片机;空气质量检测采用PM2.5传感器GP2Y1051AU0F,通过串口把检测结果发给单片机;按键部分使用市面上常见的轻触按键作为系统的输入设置模块;显示设备为1602液晶;报警则采用蜂鸣器+LED的形式;电源供电则采用USB 5V供电。

系统框图
在这里插入图片描述
原理图:
在这里插入图片描述
仿真图:
在这里插入图片描述

DHT11传感器电路
3.3.1 DHT11简介
◆相对湿度和温度测量
◆全部校准,数字输出
◆卓越的长期稳定性
◆无需额外部件
◆超长的信号传输距离
◆超低能耗
◆4 引脚安装
◆完全互换
(1)DHT11产品概述
DHT11(图3-4)数字温湿度传感器是一款含有已校准数字信号输出的温湿度复合传感器。它应用专用的数字模块采集技术和温湿度传感技术,确保产品具有极高的可靠性与卓越的长期稳定性。传感器包括一个电阻式感湿元件和一个NTC测温元件,并与一个高性能8位单片机相连接。因此该产品具有品质卓越、超快响应、抗干扰能力强、性价比极高等优点。每个DHT11传感器都在极为精确的湿度校验室中进行校准。校准系数以程序的形式储存在OTP内存中,传感器内部在检测信号的处理过程中要调用这些校准系数。单线制串行接口,使系统集成变得简易快捷。超小的体积、极低的功耗,信号传输距离可达20米以上,使其成为各类应用甚至最为苛刻的应用场合的最佳选则。产品为 4 针单排引脚封装。连接方便,特殊封装形式可根据用户需求而提供。

接口说明
(1)接口说明
建议连接线长度短于20米时用5K上拉电阻,大于20米时根据实际情况使用合适的上拉电阻。
在这里插入图片描述
PM2.5传感器模块
3.4.1 简介
日本夏普公司灰尘传感器GP2Y1051AU0F,体积小巧,灵敏度高,可以用来测量0.03微米以上的微小粒子,可用于室内环境中烟气、粉尘、花粉等浓度的检测。此款产品不但可以检测出单位体积粒子的绝对个数,而且内置气流发生器,可以自行吸入外部空气。灰尘传感器GP2Y1051AU0F安装保养方便,使用寿命长,精度高,稳定性好。其内部对角安放着红外线发光二极管和光电晶体管,使得其能够探测到空气中尘埃反射光,即使非常细小的如烟草烟雾颗粒也能够被检测到,通常在空气净化系统中应用。
3.4.2 原理说明
GP2Y1051AU0F的结构示意图如图3-11所示。首先由LED光源发射出光线,沿着LED光轴方向射出,如果空气洁净的情况下,光电二极管PD是收不到LED发出的光线的,因此输出的电压值很低。而如果空气中存在有灰尘颗粒,当这些灰尘颗粒通过传感器中间的圆孔时,由于散射的原因,LED光源发出的光线就有一部分被光电二极管PD检测到,因此就有电压值输出,并且随着灰尘颗粒浓度的升高,散射的光线越多,传感器最终输出的电压值也就越高

按键输入模块
键盘是人与单片机打交道的主要设备。站在系统监控软件设计的立场上来看,仅仅完成键盘扫描,读取当前时刻的键盘状态是不够的,还有不少问题需要妥善解决,否则,人们在操作键盘就容易引起误操作和操作失控现象。在单片机应用中键盘用得最多的形式是独立键盘及矩阵键盘。它们各有自己的特点,其中独立键盘硬件电路简单,而且在程序设计上也不复杂,一般用在对硬件电路要求不高的简单电路中;矩阵键盘与独立键盘有很大区别,首先在硬件电路上它要比独立键盘复杂得多,而且在程序算法上比它要烦琐,但它在节省端口资源上有优势得多,因此它更适合于多按键电路。其次就是消除在按键过程中产生的“毛刺”现象。这里采用最常用的方法,即延时重复扫描法,延时法的原理为:因为“毛刺”脉冲一般持续时间短,约为几ms,而我们按键的时间一般远远大于这个时间,所以当单片机检测到有按键动静后再延时一段时间后再判断此电平是否保持原状态,如果是则为有效按键,否则无效。
本设计中由于采用的按键数量较少,只有3个按键,分别是“设置”、“减”、“加”,故采用了独立键盘的方式。按键的连接图3-17所示:
在这里插入图片描述

程序设计

在这里插入图片描述

#include <reg52.h>
#include <intrins.h>

#define uchar unsigned char		// 以后unsigned char就可以用uchar代替
#define uint  unsigned int		// 以后unsigned int 就可以用uint 代替

sfr ISP_DATA  = 0xe2;			// 数据寄存器
sfr ISP_ADDRH = 0xe3;			// 地址寄存器高八位
sfr ISP_ADDRL = 0xe4;			// 地址寄存器低八位
sfr ISP_CMD   = 0xe5;			// 命令寄存器
sfr ISP_TRIG  = 0xe6;			// 命令触发寄存器
sfr ISP_CONTR = 0xe7;			// 命令寄存器

sbit LcdRs_P   = P1^2;    // 1602液晶的RS管脚       
sbit LcdRw_P   = P1^3;    // 1602液晶的RW管脚 
sbit LcdEn_P   = P1^4;    // 1602液晶的EN管脚
sbit KeySet_P  = P3^2;		// “设置”按键的管脚
sbit KeyDown_P = P3^3;		// “减”按键的管脚
sbit KeyUp_P   = P3^4;		// “加”按键的管脚 
sbit Buzzer_P  = P1^5;		// 蜂鸣器
sbit DHT11_P   = P1^1;	 	// 温湿度传感器DHT11数据接入
sbit LedTH_P   = P2^0;		// 温度过高报警指示灯
sbit LedTL_P   = P2^1;		// 温度过低报警指示灯
sbit LedHH_P   = P2^2;		// 湿度过高报警指示灯
sbit LedHL_P   = P2^3;		// 湿度过低报警指示灯
sbit LedPM_P   = P2^4;		// PM2.5过高报警指示灯

uchar temp;								// 保存温度
uchar humi;								// 保存湿度
uint  pm;									// 保存PM2.5

uchar gIndex=0;						// 串口接收索引
uint  Value[20]={0};			// 串口数据缓存区

uchar AlarmTL;						// 温度下限报警值
uchar AlarmTH;						// 温度上限报警值
uchar AlarmHL; 						// 湿度下限报警值
uchar AlarmHH;						// 湿度上限报警值
uint  AlarmPM;						// PM2.5报警值



/*********************************************************/
// 单片机内部EEPROM不使能
/*********************************************************/
void ISP_Disable()
{
	ISP_CONTR = 0;
	ISP_ADDRH = 0;
	ISP_ADDRL = 0;
}


/*********************************************************/
// 从单片机内部EEPROM读一个字节,从0x2000地址开始
/*********************************************************/
unsigned char EEPROM_Read(unsigned int add)
{
	ISP_DATA  = 0x00;
	ISP_CONTR = 0x83;
	ISP_CMD   = 0x01;
	ISP_ADDRH = (unsigned char)(add>>8);
	ISP_ADDRL = (unsigned char)(add&0xff);
	// 对STC89C51系列来说,每次要写入0x46,再写入0xB9,ISP/IAP才会生效
	ISP_TRIG  = 0x46;	   
	ISP_TRIG  = 0xB9;
	_nop_();
	ISP_Disable();
	return (ISP_DATA);
}


/*********************************************************/
// 往单片机内部EEPROM写一个字节,从0x2000地址开始
/*********************************************************/
void EEPROM_Write(unsigned int add,unsigned char ch)
{
	ISP_CONTR = 0x83;
	ISP_CMD   = 0x02;
	ISP_ADDRH = (unsigned char)(add>>8);
	ISP_ADDRL = (unsigned char)(add&0xff);
	ISP_DATA  = ch;
	ISP_TRIG  = 0x46;
	ISP_TRIG  = 0xB9;
	_nop_();
	ISP_Disable();
}


/*********************************************************/
// 擦除单片机内部EEPROM的一个扇区
// 写8个扇区中随便一个的地址,便擦除该扇区,写入前要先擦除
/*********************************************************/
void Sector_Erase(unsigned int add)	  
{
	ISP_CONTR = 0x83;
	ISP_CMD   = 0x03;
	ISP_ADDRH = (unsigned char)(add>>8);
	ISP_ADDRL = (unsigned char)(add&0xff);
	ISP_TRIG  = 0x46;
	ISP_TRIG  = 0xB9;
	_nop_();
	ISP_Disable();
}


/*********************************************************/
// 毫秒级的延时函数,time是要延时的毫秒数
/*********************************************************/
void DelayMs(uint time)
{
	uint i,j;
	for(i=0;i<time;i++)
		for(j=0;j<112;j++);
}


/*********************************************************/
// 10us级延时程序
/*********************************************************/
void Delay10us()
{
	_nop_();	// 执行一条指令,延时1微秒
	_nop_();
	_nop_();
	_nop_();
	_nop_();
	_nop_();
	_nop_();
	_nop_();
	_nop_();
	_nop_();
}


/*********************************************************/
// 读取DHT11单总线上的一个字节
/*********************************************************/
uchar DhtReadByte(void)
{   
	bit bit_i; 
	uchar j;
	uchar dat=0;

	for(j=0;j<8;j++)    
	{
		while(!DHT11_P);	// 等待低电平结束	
		Delay10us();			// 延时
		Delay10us();
		Delay10us();
		if(DHT11_P==1)		// 判断数据线是高电平还是低电平
		{
			bit_i=1; 
			while(DHT11_P);
		} 
		else
		{
			bit_i=0;
		}
		dat<<=1;		   		// 将该位移位保存到dat变量中
		dat|=bit_i;    
	}
	return(dat);  
}


/*********************************************************/
// 读取DHT11的一帧数据,湿高、湿低(0)、温高、温低(0)、校验码
/*********************************************************/
void ReadDhtData()
{    	 
	uchar HumiHig;		// 湿度高检测值
	uchar HumiLow;		// 湿度低检测值 
	uchar TemHig;			// 温度高检测值
	uchar TemLow;			// 温度低检测值
	uchar check;			// 校验字节 
	
	DHT11_P=0;				// 主机拉低
	DelayMs(20);			// 保持20毫秒
	DHT11_P=1;				// DATA总线由上拉电阻拉高

	Delay10us();	 		// 延时等待30us
	Delay10us();
	Delay10us();

	while(!DHT11_P);	// 等待DHT的低电平结束
	while(DHT11_P);		// 等待DHT的高电平结束

	//进入数据接收状态
	HumiHig = DhtReadByte(); 	// 湿度高8位
	HumiLow = DhtReadByte(); 	// 湿度低8为,总为0
	TemHig  = DhtReadByte(); 	// 温度高8位 
	TemLow  = DhtReadByte(); 	// 温度低8为,总为0 
	check   = DhtReadByte();	// 8位校验码,其值等于读出的四个字节相加之和的低8位

	DHT11_P=1;				// 拉高总线

	if(check==HumiHig + HumiLow + TemHig + TemLow) 		// 如果收到的数据无误
	{
		temp=TemHig; 			// 将温度的检测结果赋值给全局变量temp
		humi=HumiHig;			// 将湿度的检测结果赋值给全局变量humi
	}
}



/*********************************************************/
// 1602液晶写命令函数,cmd就是要写入的命令
/*********************************************************/
void LcdWriteCmd(uchar cmd)
{ 
	LcdRs_P = 0;
	LcdRw_P = 0;
	LcdEn_P = 0;
	P0=cmd;
	DelayMs(2);
	LcdEn_P = 1;    
	DelayMs(2);
	LcdEn_P = 0;	
}


/*********************************************************/
// 1602液晶写数据函数,dat就是要写入的数据
/*********************************************************/
void LcdWriteData(uchar dat)
{
	LcdRs_P = 1; 
	LcdRw_P = 0;
	LcdEn_P = 0;
	P0=dat;
	DelayMs(2);
	LcdEn_P = 1;    
	DelayMs(2);
	LcdEn_P = 0;
}


/*********************************************************/
// 1602液晶初始化函数
/*********************************************************/
void LcdInit()
{
	LcdWriteCmd(0x38);        // 16*2显示,5*7点阵,8位数据口
	LcdWriteCmd(0x0C);        // 开显示,不显示光标
	LcdWriteCmd(0x06);        // 地址加1,当写入数据后光标右移
	LcdWriteCmd(0x01);        // 清屏
}


/*********************************************************/
// 液晶光标定位函数
/*********************************************************/
void LcdGotoXY(uchar line,uchar column)
{
	// 第一行
	if(line==0)        
		LcdWriteCmd(0x80+column); 
	 // 第二行
	if(line==1)        
		LcdWriteCmd(0x80+0x40+column); 
}


/*********************************************************/
// 液晶输出数字(PM2.5的)
/*********************************************************/
void LcdPrintNum1(uint num)
{
	LcdWriteData(num/100+48);					// 百位
	LcdWriteData(num%100/10+48);			// 十位
	LcdWriteData(num%10+48); 					// 个位
}


/*********************************************************/
// 液晶输出数字(温湿度的)
/*********************************************************/
void LcdPrintNum2(uchar num)
{
	LcdWriteData(num/10+48);					// 十位
	LcdWriteData(num%10+48); 					// 个位
}



/*********************************************************/
// 液晶输出字符串函数
/*********************************************************/
void LcdPrintStr(uchar *str)
{
	while(*str!='\0')
		LcdWriteData(*str++);
}


/*********************************************************/
// 液晶显示内容初始化
/*********************************************************/
void LcdShowInit()
{
	LcdGotoXY(0,0);										// 液晶光标定位到第0行第0列
	LcdPrintStr(" PM2.5:   ug/m3 ");	// 显示内容
	LcdGotoXY(1,0);										// 液晶光标定位到第1行第0列
	LcdPrintStr("T:   C   H:  %RH");	// 显示内容
	LcdGotoXY(1,4);										// 温度单位摄氏度上面的圆圈符号
	LcdWriteData(0xdf);	
}


/*********************************************************/
// 串口初始化
/*********************************************************/
void UartInit()
{
	SCON = 0x50;		// 配置串口寄存器
	TMOD = 0x20;		// 配置定时器寄存器
	TH1  = 0xf4;		// 计算波特率的值为2400
	TL1  = 0xf4;		// 计算波特率的值为2400
	EA   = 1;				// 打开总中断
	ES   = 1;				// 打开串口中断
	TR1  = 1;				// 启动定时器
}


/*********************************************************/
// 按键扫描
/*********************************************************/
void KeyScanf()
{
	if(KeySet_P==0)		// 判断设置按键是否被按下
	{
		EA=0;
		
		/*将液晶显示改为设置温度的页面****************************************************/
		LcdWriteCmd(0x01);				  	
		LcdGotoXY(0,0);
		LcdPrintStr("Temperature Set ");
		LcdGotoXY(1,0);
		LcdPrintStr("      -    C    ");
		LcdGotoXY(1,10);	 					
		LcdWriteData(0xdf);			

		LcdGotoXY(1,4);	 					// 在液晶上填充温度的下限值	
		LcdPrintNum2(AlarmTL);	
		LcdGotoXY(1,7);	 					// 在液晶上填充温度的上限值
		LcdPrintNum2(AlarmTH);

		LcdGotoXY(1,5);	 					// 光标定位到第1行第5列
		LcdWriteCmd(0x0f);				// 光标闪烁
		
		DelayMs(10);	  					// 去除按键按下的抖动
		while(!KeySet_P);	 				// 等待按键释放
		DelayMs(10);					  	// 去除按键松开的抖动

		/*设置温度的下限值****************************************************************/
		while(KeySet_P)						// “设置键”没有被按下,则一直处于温度下限的设置
		{
			if(KeyDown_P==0)				// 判断 “减按键“ 是否被按下		
			{
				if(AlarmTL>0)					// 只有当温度下限值大于0时,才能减1
					AlarmTL--;
				LcdGotoXY(1,4);	 			// 重新刷新显示更改后的温度下限值	
				LcdPrintNum2(AlarmTL);  		
				LcdGotoXY(1,5);				// 重新定位闪烁的光标位置
				DelayMs(350);					// 延时
			}
			if(KeyUp_P==0)		  		// 判断 “加按键“ 是否被按下
			{
				if(AlarmTL<99)	   		// 只有当温度下限值小于99时,才能加1
					AlarmTL++;
				LcdGotoXY(1,4);	 	 		// 重新刷新显示更改后的温度下限值
				LcdPrintNum2(AlarmTL);
				LcdGotoXY(1,5);				// 重新定位闪烁的光标位置
				DelayMs(350);					// 延时
			}	
		}

		LcdGotoXY(1,8);
		DelayMs(10);	  					// 去除按键按下的抖动
		while(!KeySet_P);	 				// 等待按键释放
		DelayMs(10);					  	// 去除按键松开的抖动

		/*设置温度的上限值****************************************************************/	
		while(KeySet_P)	  				// “设置键”没有被按下,则一直处于温度上限的设置
		{
			if(KeyDown_P==0)				// 判断 “减按键“ 是否被按下
			{
				if(AlarmTH>0)  				// 只有当温度上限值大于0时,才能减1			
					AlarmTH--;
				LcdGotoXY(1,7);	 	  	// 重新刷新显示更改后的温度上限值
				LcdPrintNum2(AlarmTH);
				LcdGotoXY(1,8);				// 重新定位闪烁的光标位置
				DelayMs(350);					// 延时
			}
			if(KeyUp_P==0)			   	// 判断 “加按键“ 是否被按下
			{
				if(AlarmTH<99)	 			// 只有当温度上限值小于99时,才能加1
					AlarmTH++;
				LcdGotoXY(1,7);				// 重新刷新显示更改后的温度上限值 	
				LcdPrintNum2(AlarmTH);
				LcdGotoXY(1,8);				// 重新定位闪烁的光标位置
				DelayMs(350);					// 延时
			}								 
		}

		/*将液晶显示改为设置湿度的页面****************************************************/
		LcdWriteCmd(0x01);				  	
		LcdGotoXY(0,0);
		LcdPrintStr("  Humidity Set  ");
		LcdGotoXY(1,0);
		LcdPrintStr("      -   %RH   ");		

		LcdGotoXY(1,4);	 					// 在液晶上填充湿度的下限值	
		LcdPrintNum2(AlarmHL);	
		LcdGotoXY(1,7);	 					// 在液晶上填充湿度的上限值
		LcdPrintNum2(AlarmHH);

		LcdGotoXY(1,5);	 					// 光标定位到第1行第5列
		
		DelayMs(10);	  					// 去除按键按下的抖动
		while(!KeySet_P);	 				// 等待按键释放
		DelayMs(10);
		
		/*设置湿度的下限值****************************************************************/
		while(KeySet_P)				 		// “设置键”没有被按下,则一直处于湿度下限的设置
		{
			if(KeyDown_P==0)				// 判断 “减按键“ 是否被按下
			{
				if(AlarmHL>0)	 				// 只有当湿度下限值大于0时,才能减1
					AlarmHL--;
				LcdGotoXY(1,4);				// 重新刷新显示更改后的湿度下限值 	
				LcdPrintNum2(AlarmHL);
				LcdGotoXY(1,5);				// 重新定位闪烁的光标位置
				DelayMs(350);
			}
			if(KeyUp_P==0)			  	// 判断 “加按键“ 是否被按下
			{
				if(AlarmHL<99)	  		// 只有当湿度下限值小于99时,才能加1
					AlarmHL++;
				LcdGotoXY(1,4);	 		 	// 重新刷新显示更改后的湿度下限值
				LcdPrintNum2(AlarmHL);
				LcdGotoXY(1,5);	  		// 重新定位闪烁的光标位置
				DelayMs(350);					// 延时
			}	
		}

		LcdGotoXY(1,8);
		DelayMs(10);	  					// 去除按键按下的抖动
		while(!KeySet_P);	 				// 等待按键释放
		DelayMs(10);					  	// 去除按键松开的抖动
		
		/*设置湿度的上限值****************************************************************/
		while(KeySet_P)				   	// “设置键”没有被按下,则一直处于湿度上限的设置
		{
			if(KeyDown_P==0)		 		// 判断 “减按键“ 是否被按下
			{
				if(AlarmHH>0)			  	// 只有当湿度上限值大于0时,才能减1
					AlarmHH--;
				LcdGotoXY(1,7);	 		 	// 重新刷新显示更改后的湿度上限值
				LcdPrintNum2(AlarmHH);
				LcdGotoXY(1,8);		   	// 重新定位闪烁的光标位置
				DelayMs(350);
			}
			if(KeyUp_P==0)				 	// 判断 “加按键“ 是否被按下
			{
				if(AlarmHH<99)				// 只有当湿度上限值小于99时,才能加1
					AlarmHH++;
				LcdGotoXY(1,7);	 			// 重新刷新显示更改后的湿度上限值	
				LcdPrintNum2(AlarmHH);
				LcdGotoXY(1,8);	 			// 重新定位闪烁的光标位置
				DelayMs(350);					// 延时
			}	
		}

		/*将液晶显示改为设置PM2.5的页面****************************************************/
		LcdWriteCmd(0x01);				  	// 设置界面的显示框架
		LcdGotoXY(0,0);
		LcdPrintStr("   PM2.5 Set    ");
		LcdGotoXY(1,0);
		LcdPrintStr("        ug/m3   ");
		LcdGotoXY(1,4);								// 显示当前的报警值
		LcdPrintNum1(AlarmPM);				

		LcdGotoXY(1,6);	 							// 光标定位到第1行第6列
		DelayMs(10);	  							// 去除按键按下的抖动
		while(!KeySet_P);	 						// 等待按键释放
		DelayMs(10);
		
		while(KeySet_P)				 				// “设置键”没有被按下,则一直处于光强下限的设置
		{
			if(KeyDown_P==0)						// 判断 “减按键“ 是否被按下
			{
				if(AlarmPM>1)							// 只有gAlarmPM大于1才能减1								
					AlarmPM--;				
				LcdGotoXY(1,4);						// 液晶光标定位
				LcdPrintNum1(AlarmPM);		// 刷新改变后的报警值
				LcdGotoXY(1,6);
				DelayMs(200);							// 延时一下
			}
			if(KeyUp_P==0)			  			// 判断 “加按键“ 是否被按下
			{
			if(AlarmPM<999)							// 只有gAlarmPM小于999才能加1
					AlarmPM++;				
				LcdGotoXY(1,4);						// 液晶光标定位
				LcdPrintNum1(AlarmPM);		// 刷新改变后的报警值
				LcdGotoXY(1,6);
				DelayMs(200);							// 延时一下
			}	
		}

		/*完成设置,退出前的处理**********************************************************/
		LcdWriteCmd(0x0C);	  						// 取消光标闪烁
		LcdShowInit();										// 液晶显示为检测界面的

		DelayMs(10);	  									// 去除按键按下的抖动
		while(!KeySet_P);	 								// 等待按键释放
		DelayMs(10);					  					// 去除按键松开的抖动

		Sector_Erase(0x2000);			 				// 存储之前必须先擦除
		EEPROM_Write(0x2000,AlarmTL);			// 把温度下限存入到EEPROM的0x2000这个地址
		EEPROM_Write(0x2001,AlarmTH);			// 把温度上限存入到EEPROM的0x2001这个地址
		EEPROM_Write(0x2002,AlarmHL);			// 把湿度下限存入到EEPROM的0x2002这个地址
		EEPROM_Write(0x2003,AlarmHH);			// 把湿度上限存入到EEPROM的0x2003这个地址
		EEPROM_Write(0x2004,AlarmPM/100);	// 把PM2.5存入到EEPROM的0x2004和0x2005这两个地址
		EEPROM_Write(0x2005,AlarmPM%100);
		
		EA=1;
	}	
}


/*********************************************************/
// 报警判断
/*********************************************************/
void AlarmJudge(void)
{
	/*温度*/
	if(temp>AlarmTH)				// 温度是否过高
	{
		LedTH_P=0;
		LedTL_P=1;
	}
	else if(temp<AlarmTL)		// 温度是否过低
	{
		LedTL_P=0;
		LedTH_P=1;
	}
	else										// 温度正常
	{
		LedTH_P=1;
		LedTL_P=1;
	}

	/*湿度*/
	if(humi>AlarmHH)	   		// 湿度是否过高
	{
		LedHH_P=0;
	  LedHL_P=1;
	}
	else if(humi<AlarmHL)		// 湿度是否过低
	{
		LedHL_P=0;
		LedHH_P=1;
	}
	else				   					// 湿度正常
	{
		LedHH_P=1;
		LedHL_P=1;
	}
	
	/*PM2.5*/
	if(pm>AlarmPM)
		LedPM_P=0;
	else
		LedPM_P=1;

	/*蜂鸣器*/
	if((LedHH_P==0)||(LedHL_P==0)||(LedTH_P==0)||(LedTL_P==0)||(LedPM_P==0)) 	// 蜂鸣器判断,只要至少1个报警灯亮,蜂鸣器就报警
		Buzzer_P=0;
	else	
		Buzzer_P=1;
}


/*********************************************************/
// 报警值初始化
/*********************************************************/
void AlarmInit(void)
{
	AlarmTL=EEPROM_Read(0x2000);	// 从EEPROM的0x2000这个地址读取温度的报警下限
	AlarmTH=EEPROM_Read(0x2001);	// 从EEPROM的0x2001这个地址读取温度的报警上限
	AlarmHL=EEPROM_Read(0x2002);	// 从EEPROM的0x2002这个地址读取湿度的报警下限	
	AlarmHH=EEPROM_Read(0x2003);	// 从EEPROM的0x2003这个地址读取湿度的报警上限
	AlarmPM=EEPROM_Read(0x2004)*100+EEPROM_Read(0x2005);		// 读取PM2.5报警值
	
	if((AlarmTL==0)||(AlarmTL>100))	// 如果温度下限报警值读出来异常(等于0或大于100),则重新赋值
		AlarmTL=20;
	if((AlarmTH==0)||(AlarmTH>100))	// 如果温度上限报警值读出来异常(等于0或大于100),则重新赋值
		AlarmTH=35;
	if((AlarmHL==0)||(AlarmHL>100))	// 如果温度下限报警值读出来异常(等于0或大于100),则重新赋值
		AlarmHL=40;
	if((AlarmHH==0)||(AlarmHH>100))	// 如果温度上限报警值读出来异常(等于0或大于100),则重新赋值
		AlarmHH=85;
	if((AlarmPM==0)||(AlarmPM>1300))	// 如果读取到的报警值异常,则重新赋值
		AlarmPM=200;
}


/*********************************************************/
// 主函数
/*********************************************************/
void main(void)
{
	uchar i;				// 循环变量
	uint ret;				// 保存PM2.5测量结果

	LcdInit();				    	// 液晶功能初始化
	LcdShowInit();					// 液晶显示初始化
	UartInit();							// 串口初始化
	AlarmInit();						// 报警值初始化

	while(1)
	{
		/*PM2.5的读取*/
		ret=0;											// 清零测量结果
		for(i=0;i<10;i++)						// 将最新的20个测量结果求和
		{
			ret=ret+Value[i];
		}
		ret=ret/10;									// 再除以20求得平均值
		pm=((ret*5)/1024.0)*180;		// 将读取到的电压值转换为灰尘浓度值	
		LcdGotoXY(0,7);							// 液晶定位到第0行第6列
		LcdPrintNum1(pm);						// 显示测量结果
		
		/*温湿度读取*/
		EA=0;
		ReadDhtData(); 							// 检测温湿度数据;
		EA=1;
		LcdGotoXY(1,2);	 						// 定位到要显示温度的地方
		LcdPrintNum2(temp);					// 显示温度值
		LcdGotoXY(1,11);						// 定位到要显示湿度的地方
		LcdPrintNum2(humi);					// 显示湿度值
		
		// 报警判断
		AlarmJudge();						
		
		/*按键扫描和延时*/
		for(i=0;i<30;i++)
		{
			KeyScanf();								// 按键判断
			DelayMs(10);
		}
	}
}


/*********************************************************/
// 串口中断服务程序
/*********************************************************/
void UartInt(void) interrupt 4
{
	uchar VoutH,VoutL;
	
	if(RI==1)
	{
		RI=0;
		if(SBUF==0xAA)		// 起始位
		{
			while(!RI);
			VoutH=SBUF;			// Vout(H)
			RI=0;
			while(!RI);
			VoutL=SBUF;			// Vout(L)
			RI=0;
			while(!RI);			// Vref(H)
			RI=0;
			while(!RI);			// Vref(L)
			RI=0;
			while(!RI);			// 校验位
			RI=0;
			while(!RI);			// 停止位
			RI=0;
			
			Value[gIndex]=VoutH*256+VoutL;
			
			gIndex++;
			if(gIndex==20)
				gIndex=0;
		}
	}
} 





.

链接:https://pan.baidu.com/s/10tTKSPlu5fRm-jH5PAXU6A
提取码:zt6j

.

  • 14
    点赞
  • 156
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值