基于51单片机的pm2.5空气质量监测仪仿真设计

news2024/12/24 9:36:14

51单片机pm2.5监测仪仿真设计( proteus仿真+程序+报告+讲解视频)

仿真图proteus 7.8及以上

程序编译器:keil 4/keil 5

编程语言:C语言

设计编号:S0032

51单片机pm2.5监测仪仿真设计

  • 主要功能:
  • 讲解演示视频
  • 仿真
  • 程序
  • 设计报告
      • 1.1.1、设计的主要内容
      • 1.1.2、设计的主要意义
  • 资料清单以及下载方式

主要功能:

1.设计要求

采用51单片机作为核心控制器,检测pm2.5浓度。

简要设计内容:

1、pm2.5的检测与显示;

2、当pm2.5>100时报警;

3、其他创新内容

具体功能:

1.通过ADC0832检测pm2.5的值,仿真通过滑动变阻器改变PM2.5数值。

2.LCD1602第一行显示PM2.5实时检测值,第二行显示报警值。

3.默认当pm2.5>100时蜂鸣器报警。

4.可以通过按键设置pm2.5的报警值。

以下为本设计资料展示图:

讲解演示视频

51单片机PM2.5空气质量监测仪proteus仿真设计

仿真

img

开始仿真

1.通过ADC0832检测pm2.5的值,仿真通过滑动变阻器改变PM2.5数值。

2.LCD1602第一行显示PM2.5实时检测值,第二行显示报警值。

3.默认当pm2.5>100时蜂鸣器报警。

4.可以通过按键设置pm2.5的报警值。

程序

img

#include <reg52.h>
#include <intrins.h>
#define uint unsigned int
#define uchar unsigned char //宏定义
sbit RS=P1^6;//液晶接口
sbit EN=P1^7;

sbit LED = P2^0;//粉尘传感器控制接口

sbit ADCS = P3^7;//AD0832接口
sbit ADCLK =P3^5;
sbit ADDI = P3^6;
sbit ADDO = P3^6;		  

sbit SET= P1^1;//按键接口
sbit ADD= P1^2;
sbit DEC= P1^3;

sbit BEEP=P2^1;//蜂鸣器接口

uchar set_st;
uchar tab[5];
uint DUST_SET=100; //固体颗粒的阈值

//bit shanshuo_st; //闪烁间隔标志
bit beep_st; //蜂鸣器间隔标志
uchar x=4; //计数器
//定义标识
uchar FlagStart = 0;
float DUST_Value;
uint DUST;
uchar num=0;
uchar mm;
uchar abc;
uchar ADC_Get[10]={0}; //定义AD采样数组
uchar str[5]={0};

/*****初始化定时器0*****/
void InitTimer(void)
{
	TMOD = 0x01;
	TL0 = (65536-10000)/256; //定时10ms
	TH0 = (65536-10000)%256;
	TR0 = 1;
	ET0 = 1;
	EA = 1;
}
/*************************lcd1602程序**************************/
void delay1ms(uint ms)//延时1毫秒
{ 
    uint i,j;
	for(i=0;i<ms;i++)
	for(j=0;j<100;j++);
}

void wr_com(uchar com)//写指令//
{ 
    delay1ms(1);
	RS=0;
//	RW=0;
	EN=0;
	P0=com;
	delay1ms(1);
	EN=1;
	delay1ms(1);
	EN=0;
}
void wr_dat(uchar dat)//写数据//
{ 
    delay1ms(1);;
	RS=1;
//	RW=0;
	EN=0;
	P0=dat;
	delay1ms(1);
	EN=1;
	delay1ms(1);
	EN=0;
}
/*****************************液晶初始化
*********************************************/
void lcd_init()//初始化设置//
{ 
	delay1ms(15);
	wr_com(0x38);
	delay1ms(5);
	wr_com(0x01);
	delay1ms(5);
	wr_com(0x06);
	delay1ms(5);
	wr_com(0x0c);
	delay1ms(5);

	wr_com(0x80);
	wr_dat('P');//
	wr_com(0x81);
	wr_dat('M');//:
	wr_com(0x82);
	wr_dat('2');//
	wr_com(0x83);
	wr_dat('.');//:
	wr_com(0x84);
	wr_dat('5');//:
	wr_com(0x85);
	wr_dat(':');


	wr_com(0x8b);
	wr_dat('u');
	wr_com(0x8c);
	wr_dat('g');
	wr_com(0x8d);
	wr_dat('/');
	wr_com(0x8e);
	wr_dat('m');
	wr_com(0x8f);
	wr_dat('3');
	


/

	wr_com(0xc0);
	wr_dat('A');
	wr_com(0xc1);
	wr_dat('l');
	wr_com(0xc2);
	wr_dat('a');
	wr_com(0xc3);
	wr_dat('r');
	wr_com(0xc4);
	wr_dat('m');
	wr_com(0xc5);
	wr_dat(':');

	wr_com(0xcb);
	wr_dat('u');
	wr_com(0xcc);
	wr_dat('g');
	wr_com(0xcd);
	wr_dat('/');
	wr_com(0xce);
	wr_dat('m');
	wr_com(0xcf);
	wr_dat('3');
}
/*****************显示函数******************************/
void disp(unsigned int Data)//PM2.5值显示
{
	uint Temp;
	Temp=Data%10000;
	str[0]=' ';//Temp/1000+0x30; //千位
	Temp%=1000;
	str[1]=' ';
	str[2]=Temp/100+0x30; //百位
	Temp%=100;
	str[3]=Temp/10+0x30; //十位
	str[4]=Temp%10+0x30; //个位
	wr_com(0x86);
	wr_dat(str[0]);
	wr_com(0x87);
	wr_dat(str[1]);
	wr_com(0x88);
	wr_dat(str[2]);
	wr_com(0x89);
	wr_dat(str[3]);
	wr_com(0x8a);
	wr_dat(str[4]);

}
/************************报警值显示************************************/ 
void baojing()
{
	wr_com(0xc6);
	wr_dat(tab[0]);
	wr_com(0xc7);
	wr_dat(tab[1]);
	wr_com(0xc8);
	wr_dat(tab[2]+0x30);
	wr_com(0xc9);
	wr_dat(tab[3]+0x30);
	wr_com(0xca);
	wr_dat(tab[4]+0x30);
}
/*****延时子程序*****/
void Delay(uint num)
{
while( --num );
}
/**************************按键检测
*******************************************/
void checkkey()
{
	if(SET==0)
	{
	Delay(2000);
	do{}while(SET==0);
	set_st++;
	if(set_st>1)set_st=0;
	}
	if(set_st==0)
	{
	}
	else if(set_st==1)
	{
	if(DEC==0)
	{
	Delay(2000);
	do{}while(DEC==0);
	if(DUST_SET>0)DUST_SET--;
	if(DUST_SET==0)DUST_SET=0;
	}
	if(ADD==0)
	{
	Delay(2000);
	do{}while(ADD==0);
	DUST_SET++;
	if(DUST_SET>800)DUST_SET=800;
	}
	}
	tab[0]=' ';//DUST_SET/1000;
	tab[1]=' ';
	tab[2]=DUST_SET%1000/100;
	tab[3]=DUST_SET%100/10;
	tab[4]=DUST_SET%10;
}
/*****报警子程序*****/
void Alarm()
{
	if(x>=10){beep_st=~beep_st;x=0;}
	if(DUST>DUST_SET&&beep_st==1)BEEP=0;
	else BEEP=1;
//	if(DUST>0&&DUST<100){LED2=0;LED3=1;LED4=1;}
//	if(DUST>=10&&DUST<300){LED2=1;LED3=0;LED4=1;}
//	if(DUST>=300){LED2=1;LED3=1;LED4=0;}
	}
/**************************AD0832转换程序
***********************************************/
uchar ADC0832(bit mode,bit channel) //AD转换,返回结果
{
	uchar i,dat,ndat;
	ADCS = 0;//拉低CS端
	_nop_();
	_nop_();
	ADDI = 1; //第1个下降沿为高电平
	ADCLK = 1;//拉高CLK端
	_nop_();
	_nop_();
	ADCLK = 0;//拉低CLK端,形成下降沿1
	_nop_();
	_nop_();
	ADDI = mode; //低电平为差分模式,高电平为单通道模式。
	ADCLK = 1;//拉高CLK端
	_nop_();
	_nop_();
	ADCLK = 0;//拉低CLK端,形成下降沿2
	_nop_();
	_nop_();
	ADDI = channel; //低电平为CH0,高电平为CH1
	ADCLK = 1;//拉高CLK端
	_nop_();
	_nop_();
	ADCLK = 0;//拉低CLK端,形成下降沿3
	ADDI = 1;//控制命令结束(经试验必需)
	dat = 0;
	//下面开始读取转换后的数据,从最高位开始依次输出(D7~D0) 
	for(i = 0;i < 8;i++)
	{
	dat <<= 1;
	ADCLK=1;//拉高时钟端
	_nop_();
	_nop_();
	ADCLK=0;//拉低时钟端形成一次时钟脉冲
	_nop_();
	_nop_();
	dat |= ADDO;
	}
	ndat = 0; //记录D0
	if(ADDO == 1)
	ndat |= 0x80;
	//下面开始继续读取反序的数据(从D1到D7)
	for(i = 0;i < 7;i++)
	{
	ndat >>= 1;
	ADCLK = 1;//拉高时钟端
	_nop_();
	_nop_();
	ADCLK=0;//拉低时钟端形成一次时钟脉冲
	_nop_();
	_nop_();
	if(ADDO==1)
	ndat |= 0x80;
	}
	ADCS=1;//拉高CS端,结束转换
	ADCLK=0;//拉低CLK端
	ADDI=1;//拉高数据端,回到初始状态
	if(dat==ndat)
	return(dat);
	else
	return 0;
}
/*****定时器0中断服务程序*****/
void timer0(void) interrupt 1
{
	uint j;
	TL0 = (65536-10000)/256; //定时10ms
	TH0 = (65536-10000)%256;
	LED=1; //开启传感器的LED
	x++;
	for (j=0;j<30;j++); //0.28ms //延时0.28ms
	abc=ADC0832(1,0); //开启ADC采集
	FlagStart=1;
	TR0 = 0; //先关闭定时器0
	EA = 0;
	LED=0;//关闭传感器LED
}
//中值滤波
//算法:先进行排序,然后将数组的中间值作为当前值返回。
uchar Error_Correct(uchar *str,uchar num)
{
	unsigned char i=0;
	unsigned char j=0;
	unsigned char Temp=0;
	//排序
	for(i=0;i<num-1;i++)
	{
	for(j=i+1;j<num;j++)
	{
	   if(str[i]<str[j])
		{
		Temp=str[i];
		str[i]=str[j];
		str[j]=Temp;
		}
	}
}

	//去除误差,取中间值
	return str[num/2];
}
/*****主函数*****/
void main(void)
{
	InitTimer(); //初始化定时器
	BEEP=1;
	lcd_init();//初始化显示
	delay1ms(500);
	while(1)
	{

	 checkkey();//按键检测
	if(set_st==0)
	{
		//wr_com(0x0c);
		if(FlagStart==1) //1次数据采集完成
		{
		num++;
		ADC_Get[num]=abc;
		if(num>9)
		{
		num=0;
//		DUST=Error_Correct(ADC_Get,10); //求取10次AD采样的值 
//		DUST_Value=(DUST/256.0)*5000; //转化成电压值MV
//		DUST_Value=DUST_Value*0.17-100; //固体悬浮颗粒浓度计算 Y=0.17*X-0.1 X--采样电压V
	
		DUST=Error_Correct(ADC_Get,10);
	    DUST_Value=(DUST/256.0)*5;//转化成电压值
	    DUST_Value=(DUST_Value*0.17-0.1)*1000;//固体悬浮颗粒浓度计
	
		if(DUST_Value<0) DUST_Value=0;
		if(DUST_Value>760) DUST_Value=760; //限位
		DUST=(uint)DUST_Value;
		}
		TL0 = (65536-10000)/256;
		TH0 = (65536-10000)%256;
		TR0 = 1; //开启定时器0
		EA = 1;
		FlagStart=0;
		}
		Alarm(); //报警检测
	}

		if(set_st==1)//报警值闪动
		{
//		wr_com(0xca);
//		wr_com(0x0d);
//		delay1ms(150);
		}
	}
}

设计报告

报告格式规范,字数13739,含各种框图。

img

1.1、设计的主要内容和意义

1.1.1、设计的主要内容

本设计提出的检测空气质量PM2.5的方案最基本的实现方法是由单片机、粉尘监测传感器、显示模块、报警模块等组成的电路, GP2Y1010AU0F粉尘传感器采集空气中PM2.5的浓度值,经过51单片机处理后,在LCD1602液晶上显示,并通过LED和蜂鸣器提示浓度状态。

1.1.2、设计的主要意义

21世纪的今天,科学技术的发展日新月异,科学技术的进步同时也带动了测量技术的发展,现代控制设备不同于以前,它们在性能和结构发生了翻天覆地的变化。我们已经进入了高速发展的信息时代,测量技术是当今社会的主流,广泛地深入到应用工程的各个领域。

因为空气质量的恶化,阴霾天气现象出现增多,危害现象加重。中国不少地区把阴霾天气现象并入雾一起作为灾害性天气预警预报。统称为“雾霾天气”。雾霾主要由PM2.5、PM10、PM0.1以及重金属镍砷铬铅等颗粒组成。在空气动力学和环境气象学中,颗粒物是按直径大小来分类的,粒径小于100微米的称为TSP(TotalSuspendedParticle),即总悬浮物颗粒;粒径小于10微米的称为PM10(PM为ParticulateMatter缩写),即可吸入颗粒物;粒径小于2.5微米的称为PM2.5,即可入肺颗粒物,它的直径仅相当于人的头发丝粗细的1/20。虽然PM2.5只是地球大气成分中含量很少的组分,但它与较粗的大气颗粒物相比,粒径小,富含大量的有毒、有害物质且在大气中的停留时间长、输送距离远,因而对人体健康和大气环境质量影响更大。

世界卫生组织发布的报告显示,无论是发达国家还是发展中国家,目前大多数城市和农村人口均遭受到颗粒物对健康的影响。高污染城市中的死亡率超出相对清洁城市的15%至20%。据统计,在欧洲,PM2.5每年导致386000人死亡,并使欧盟国家人均期望寿命减少8.6个月。人体的生理结构决定了对PM2.5没有任何过滤、阻拦能力,而PM2.5对人类健康的危害却随着医学技术的进步,逐步暴露出其恐怖的一面。气象专家和医学专家认为,由细颗粒物造成的灰霾天气对人体健康的危害甚至要比沙尘暴更大。粒径10微米以上的颗粒物,会被挡在人的鼻子外面;粒径在2.5微米至10微米之间的颗粒物,能够进入上呼吸道,但部分可通过痰液等排出体外,另外也会被鼻腔内部的绒毛阻挡,对人体健康危害相对较小;而粒径在2.5微米以下的细颗粒物,直径相当于人类头发的1/10大小,不易被阻挡。被吸入人体后会直接进入支气管,刺激呼吸道,干扰肺部的气体交换,从而引发咳嗽、呼吸困难、哮喘、慢性支气管炎等呼吸系统的疾病并导致心律不齐、非致命性心脏病等心血管方面的疾病。其中,老人、小孩以及心肺疾病患者是PM2.5污染的敏感人群。

因此,对PM2.5的监测与治理便显得越来越重要。

资料清单以及下载方式

见视频
img

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/165545.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

代码整洁提升方案

验-言 公共方法都要做参数的校验&#xff0c;参数校验不通过明确抛出异常或对应响应码&#xff1a; Java Bean验证已经是一个很古老的技术了&#xff0c; 会避免我们很多问题&#xff1b; 在接口中也明确使用验证注解修饰参数和返回值&#xff0c; 作为一种协议要求调用方按…

win11 arm 系统安装安卓子系统

一般的x86电脑如果安装android子系统&#xff0c;运行安卓子系统&#xff0c;由于要将android arm代码转译为x86代码&#xff0c;所以效率不一定高&#xff0c;但是如果电脑是arm架构的&#xff0c;通过安卓子系统运行android的程序执行效率就会 高不少&#xff0c;本文参考,都…

JVM面试题

Java内存区域 说一下 JVM 的主要组成部分及其作用&#xff1f; JVM包含两个子系统和两个组件&#xff0c;两个子系统为Class loader(类装载)、Execution engine(执行引擎)&#xff1b;两个组件为Runtime data area(运行时数据区)、Native Interface(本地接口)。 Class loader…

Es6的Promise

Promise是异步编程的一种解决方案。简单来说Promise就是一个用来存储数据的对象&#xff0c;它有着一套特殊的存取数据的方式。可以解决异步回调函数/回调地狱问题。创建Promise1.创建Promise时需要一个 回调函数 作为参数这个回调函数会在Promise时 自动调用2.调用回调函数时&…

基于matlab的指纹图像处理、脊线增强、脊线分割、脊线细化、细节点检测和细节点验证

需求分析对于指纹的特征提取包含几个步骤&#xff0c;脊线增强、脊线分割、脊线细化、细节点检测和细节点验证&#xff0c;本次大作业需要针对已经增强的指纹图片进行后续几个步骤&#xff0c;通过多种形态学算法进行分割、细化、细化后处理&#xff0c;找到其中的端点和分叉点…

elasticsearch基础2——es配置文件、jvm配置文件详解

文章目录一、配置文件详解1.1 elasticsearch.yml文件1.1. 1 基础参数1.1.1.1 自定义数据/日志目录1.1.1.2 锁定物理内存1.1.1.3 跨域设置1.1.1.4 其他参数1.1.2 集群类1.1.3 分片类1.1.4 IP绑定类1.1.5 端口类1.1.6 交互类1.1.5 Xpcak安全认证1.1.5.1 xpack内置用户1.1.5.2 xpa…

LabVIEW使用VI脚本向VI添加对象

LabVIEW使用VI脚本向VI添加对象可使用VI脚本向前面板和程序框图添加对象。该教程以向程序框图添加对象为例。按照下列步骤&#xff0c;通过VI脚本向VI添加对象。创建VI前&#xff0c;需先了解VI脚本的基本内容。必须启用VI脚本&#xff0c;才能显示VI脚本选板&#xff0c;使用相…

aws beanstalk 理解和使用eb工作线程环境

参考资料 beanstalk 工作线程环境beanstalk 工作线程环境管理https://catalog.us-east-1.prod.workshops.aws/workshops/b317e4f5-cb38-4587-afb1-2f75be25b2c0/en-US/03-provisionresources 理解 beanstalk 工作线程环境 https://docs.amazonaws.cn/elasticbeanstalk/latest…

【Java基础】-【线程】

文章目录创建线程的方式Thread类的常用方法run()和start()有什么区别&#xff1f;线程是否可以重复启动&#xff0c;有什么后果&#xff1f;线程的生命周期实现线程同步Java多线程之间的通信方式sleep()和wait()的区别notify()、notifyAll()的区别如何实现子线程先执行&#xf…

GaussDB(DWS)数据库的数据迁移实操【玩转PB级数仓GaussDB(DWS)】

GaussDB(DWS)数据库的数据迁移实操【玩转PB级数仓GaussDB(DWS)】 1.1先了解一下Gauss数据库 Gauss数据库并非完全自主开发。它可以看作是基于PostgreSQL 9.2的一次神奇的修改。正如Redhat和Android都源于LINUX的研发&#xff0c;IBM AIX和IOS都源于UNIX的研发一样&#xff0c;…

16、ThingsBoard-配置OAuth2

1、概述 如果你的系统想要接入第三方认证来登录,就像国内很多网站都支持微信、QQ等授权登录,其实thingsboard也提供了OAuth2.0来支持,ThingsBoard 是支持授权码授权类型来交换访问令牌的授权码,同时它自己也提供了几种方式 Google、GitHub、Facebook、Apple 同时也支持自定…

使用numpy进行深度学习代码实战

使用方法定义网络from net import ConvNet net ConvNet() if not net.load(MODEL_PATH): net.addConvLayout([3,3,1,4],bias True,paddingVAILD,init_typeinit_type,st_funcLEAKY_RELU_0.01)net.addConvLayout([3,3,4,8],bias True,paddingVAILD,init_typeinit_type,st…

weblogic反序列化之T3协议

前言 weblogic 的反序列化漏洞分为两种 &#xff0c;一种是基于T3 协议的反序列化漏洞&#xff0c;一个是基于XML的反序列化漏洞&#xff0c;这篇来分析一下基于T3 协议的反序列化漏洞。 环境搭建&#xff1a; [JAVA安全]weblogic反序列化介绍及环境搭建_snowlyzz的博客-CSDN…

Virtualbox设置固定IP

Virtualbox桥接实现静态固定IP内外网访问 super_kancy 2018-10-20 11:55:28 6024 收藏 7 展开 桥接实现静态固定IP内外网访问 第一步、安装好一个虚拟机linux01 第二步、配置网卡&#xff0c;选择桥接网卡模式&#xff0c;并且指定桥接的具体的网卡 第三步、正常启动虚拟机lin…

【Java集合】Collection 体系集合详解(ArrayList,LinkedList,HashSet,TreeSet...)

文章目录1. 概念2. 集合和数组的区别3. 集合的体系结构4. Collection父接口5. List 子接口6. List 实现类6.1 ArrayList 类6.2 Vector 类6.3 LinkedList 类6.4 ArrayList和LinkedList的区别7. Set 子接口8. Set 实现类8.1 HashSet 类8.2 TreeSet 类9. Collections 工具类Java编…

【链表经典题目】总结篇

【链表经典题目】总结篇1 虚拟头结点2 链表的基本操作3 反转链表4 删除倒数第N个节点5 链表相交6 环形链表总结【链表】关于链表&#xff0c;你该了解这些&#xff01; 1 虚拟头结点 在链表&#xff1a;听说用虚拟头节点会方便很多&#xff1f; 中&#xff0c;我们讲解了链表…

简单了解OSI网络模型

本文为学习笔记&#xff0c;根据了解需求摘抄自下篇文章 参考&#xff1a;原文地址 作者&#xff1a;sunsky303 目录 OSI模型 TCP/IP分层模型 OSI模型 OSI 模型(Open System Interconnection model)&#xff08;七层网络模型&#xff09;是一个由国际标准化组织提出的概念模…

职责链模式

职责链模式 1.职责链模式基本介绍 职责链模式&#xff08;Chain of Responsibility Pattern&#xff09;, 又叫 责任链模式&#xff0c;为请求创建了一个接收者对象的链(简单示意图)。这种模式对请求的发送者和接收者进行解耦。 职责链模式通常每个接收者都包含对另一个接收者…

谷歌搜索引擎排名规则(谷歌 seo 外链重要还是内容重要)

谷歌外链仍然是Google排名前三的因素之一&#xff0c;这意味着你根本不能忽视外链带来的排名。如果不建立高质量的链接&#xff0c;现实情况是&#xff0c;你的竞争性关键字和搜索词不会有高排名的。 并非所有外链都是平等的。事实上&#xff0c;错误类型的链接可能会损害您的…

一次线上事故,我顿悟了MongoDB的精髓

目录MongoDB拒绝连接&#xff1f;显然是MongoDB服务又挂了。mongodb启动异常&#xff1a;about to fork child process, waiting until server is ready for connection一、什么是MongoDB分片&#xff1f;二、MongoDB如何分片?三、何时分片&#xff1f;四、搭建MongoDB分片服务…