环境监测系统--------MQ系列气体检测模块驱动教程(保姆级教程)


⏩ 大家好哇!我是小光,嵌入式爱好者,一个想要成为系统架构师的大三学生。
⏩在环境检测中我们经常会用到检测气体的传感器,检测乙醇、甲烷、一氧化碳、氢气等等,博主呕心沥血对MQ系列传感器做一个史上最详细的使用教程。
⏩感谢你的阅读,不对的地方欢迎指正。
加入小光嵌入式交流群(qq群号:737327353)免费获取博主所有资料哦!


MQ系列

  • 传感器说明
    • 传感器原理
    • 读取传感器数据原理
  • 硬件连接
  • 软件驱动代码
    • ADC驱动代码
    • 模块驱动代码
    • 主函数调用
  • 总结

传感器说明

传感器原理

在这里插入图片描述
MQ气体传感器使用的气敏材料是在清洁空气中电导率较低的二氧化锡(Sno2)。当传感器所处环境中存在可燃气体时,传感器的电导率随空气中可燃气体浓度的增加而增大。使用简单的电路即可将电导率的变化转换为与该气你浓度相对应的输出信号。MQ气体传感器对甲烷的灵敏度高,对丙烷、丁烷也有较好的灵敏度。这种传感器可检测多种可燃性气体,特别是天然气,是一款适合多种营养的低成本传感器。
目前有以下传感器:
在这里插入图片描述
本文包含MQ3、MQ5、MQ7、MQ135传感器的代码。
在这里插入图片描述
电器性能
输入电压:DC5V功耗(电流):150mA
DO输出:TTL数字量0和1(0.1和5V)
AO输出:0.1-0.3V(相对无污染),高浓度电压4V左右
特别提醒:传感器通电后,需要预热20S左右,测量的数据才稳定,传感器发热属于正常现象,因为内部有电热丝。

读取传感器数据原理

MQ-3 气敏元件的结构和外形如下图所示:
在这里插入图片描述
在这里插入图片描述

Rs/R0~ppm特性曲线如下图:
在这里插入图片描述
在这里插入图片描述

Rs/R0~温湿度 特性曲线
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
a=0.5447,b=-0.6785
注意:RL的电阻值是自己测量的,R0的值是根据传感器所处环境测量出的电压进行计算的,这两个变量也是代码中需要更改的,如果使用的。

硬件连接

开发板:STM32F103C8T6最小系统板
传感器:MQ3、MQ5、MQ7、MQ135

/******************引脚接口**************************
MQ3    AO	-PA1	
MQ5    AO	-PA4
MQ7    AO	-PA5
MQ135  AO	-PA7
VCC         -3.3V
GND         -开发板GND
************************************************/

软件驱动代码

ADC驱动代码

ADC模数转换就在这里不详细解答啦,如果使用的引脚不同需要在adc.h中更改相关的定义。
adc.h

#ifndef __ADC_H
#define __ADC_H	
#include "sys.h"							  

//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//STM32F103最小系统板
//MQ传感器驱动代码	   
//技术交流群:737327353
//修改日期:2024/4/21
//版本:V1.0
//版权所有,盗版必究。
//Copyright(C) CSDN 小光学嵌入式					  
/
//对应的ADC通道
#define MQ3_adc_channel ADC_Channel_1
#define MQ5_adc_channel ADC_Channel_4
#define MQ7_adc_channel ADC_Channel_5
#define MQ135_adc_channel ADC_Channel_7
//对应引脚
#define MQ3_Port GPIO_Pin_1
#define MQ5_Port GPIO_Pin_4
#define MQ7_Port GPIO_Pin_5
#define MQ135_Port GPIO_Pin_7

#define MQ_GPIOX GPIOA
void Adc_Init(void);
u16  Get_Adc(u8 ch);
u16 Get_Adc_Average(u8 ch,u8 times);

#endif 

adc.c

#include  "adc.h"
#include  "delay.h"




//初始化ADC
//这里我们仅以规则通道为例
//我们默认将开启通道0~3																	   
void  Adc_Init(void)
{
	ADC_InitTypeDef ADC_InitStructure;
	GPIO_InitTypeDef GPIO_InitStruture;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_ADC1,ENABLE);//使能ADC1通道时钟
	
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);//shezhiADC分频因子 
	
	//PA1 作为模拟通道输入引脚   
	GPIO_InitStruture.GPIO_Mode = GPIO_Mode_AIN;
	GPIO_InitStruture.GPIO_Pin = MQ3_Port|MQ5_Port|MQ7_Port|MQ135_Port;
	
	GPIO_Init(MQ_GPIOX,&GPIO_InitStruture);
	
	ADC_DeInit(ADC1);//复位ADC1
	
	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//是否持续扫描
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//数据右对齐模式
	ADC_InitStructure.ADC_ExternalTrigConv =ADC_ExternalTrigConv_None;//外部中断
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//独立模式;ADC1和ADC2独立
	ADC_InitStructure.ADC_NbrOfChannel = 5;
	ADC_InitStructure.ADC_ScanConvMode= DISABLE;//ADC转换单通道模式
	ADC_Init(ADC1, &ADC_InitStructure);	//根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器   
	
	ADC_Cmd(ADC1, ENABLE);	//使能指定的ADC1
	
	ADC_ResetCalibration(ADC1);	//使能复位校准  
	 
	while(ADC_GetResetCalibrationStatus(ADC1));	//等待复位校准结束
	
	ADC_StartCalibration(ADC1);	 //开启AD校准
 
	while(ADC_GetCalibrationStatus(ADC1));	 //等待校准结束
}
u16  Get_Adc(u8 ch)
{
	//设置指定ADC的规则组通道,一个序列,采样时间
	ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5);//ADC1,ADC通道,采样通道数量,采样周期
	
	ADC_SoftwareStartConvCmd(ADC1 , ENABLE);//使能指定的ADC1软件转换启动功能
	
	while(!ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC));//判断是否转换完成
	
	return ADC_GetConversionValue(ADC1);//返回最近一次ADC1规则组的转换结果
}
u16 Get_Adc_Average(u8 ch,u8 times)
{
	u32 temp_val=0;
	u8 t;
	for(t=0;t<times;t++)
	{
		temp_val+=Get_Adc(ch);
	}
	return temp_val/times;
}

模块驱动代码

以下都是博主自己编写的驱动程序哦,
注意:RL的电阻值是自己测量的,R0的值是根据传感器所处环境测量出的Vout电压进行计算的。
mq.h

#ifndef _MQ_H
#define _MQ_H
#include "stm32f10x.h"
#include "sys.h"
#include "adc.h"
#include "usart.h"
#include <math.h>
//MQ3 乙醇
//MQ5 甲烷
//MQ7 CO
//MQ135 氢气

typedef struct
{
	int   ID;
	double adcx;
	double Vout;
	double Rs;
	double R0;
	
}MQ;

//各个气体传感器的可调负载电阻的电阻值RL(通过万用表测量A-H端得)
//单位:KΩ
#define MQ3_RL   0.43
#define MQ5_RL   2.15
#define MQ7_RL   2.55
//#define MQ8_RL   2.55
#define MQ135_RL 2.59

//各个气体传感器的在洁净空气中的电阻值R0(通过2、3、4公式计算得)
//单位:KΩ
#define MQ3_R0   0.8369
#define MQ5_R0   4.30
#define MQ7_R0   5.0375
//#define MQ8_R0   145
#define MQ135_R0 3.3760

#define 

//功能:将获取到的AO值转化为传感器电阻值
//adcx:ADC获取的值 RL:传感器负载电阻值
double adc_to_R(double Vout,double RL);

//测量ADC的值并转化成电阻打印到串口一上
//MQx:传感器型号 ADC_Channel_x:选用的ADC通道x MQx_RL:传感器可调负载电阻值
void MQ_printf(MQ* MQx,u8 ADC_Channel_x,double MQx_RL,double MQx_R0);
#endif
#include "mq.h"


double mq3_CH3OH,mq5_CH4,mq7_CO,mq135_H2;
double mq;
//功能:将获取到的AO值转化为传感器电阻值
//adcx:ADC获取的值 RL:传感器负载电阻值
double adc_to_R(double Vout,double RL)
{
  double temp = 0.0;	
	double Rs = 0.0;
	
	Rs = (3.3 - Vout)/Vout * RL;//敏感体电阻值
	return Rs;
}

//测量ADC的值并转化成电阻打印到串口一上
//MQx:传感器型号 ADC_Channel_x:选用的ADC通道x MQx_RL:传感器可调负载电阻值
void MQ_printf(MQ* MQx,u8 ADC_Channel_x,double MQx_RL,double MQx_R0)
{
	u8 txbuffer[100];
	double ppm = 0.0;
	double x = 0.0,y = 0.0;
	MQx->adcx = Get_Adc_Average(ADC_Channel_x,10);    //从ADC通道x获取ADC的值
	MQx->Vout = MQx->adcx*(3.3/4096);
//	printf("Vout:%f\r\n",MQx->Vout);
	MQx->Rs = adc_to_R(MQx->Vout,MQ3_RL) ;             //通过获取的ADC值求传感器电阻值
	x = MQx->Rs/MQx_R0;
	switch(MQx->ID)
	{
		case 3:mq3_CH3OH=ppm = pow((0.5447/x),1.0/0.6785);
		break;
		case 5:mq5_CH4=ppm = pow(43.8499*x,-1.7381);
		break;
		case 7:mq7_CO=ppm = pow(98.3224*x,-1.4583);
		break;
		case 8:
		break;
		case 135:mq135_H2=ppm = pow(3.5314*x,-2.0437);
		break;
	}
	mq = ppm;
	sprintf((char *)txbuffer,"MQ%d_ppm:%.5f",MQx->ID, ppm);
	printf("%s",txbuffer);
	if(MQx->ID == 3) printf("mg/L\r\n\r\n");
	else printf("ppm\r\n\r\n");
}

主函数调用

main.c

#include "delay.h"
#include "sys.h"
#include "usart.h"
  #include "mq.h"
 #include "adc.h"
 #include "led.h"
//c库
#include "string.h"

//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//STM32F103最小系统板
//MQ传感器驱动代码	   
//技术交流群:737327353
//修改日期:2024/4/21
//版本:V1.0
//版权所有,盗版必究。
//Copyright(C) CSDN 小光学嵌入式					  
/
/******************引脚接口**************************
MQ3AO	PA1	
MQ5AO	PA4
MQ7AO	PA5
MQ135AO	PA7
************************************************/

 int main(void)
 {		
	u16 timeCount=0;
	 MQ MQ3,MQ5,MQ7,MQ8,MQ135;
	
	 MQ3.ID = 3;
	 MQ5.ID = 5;
	 MQ7.ID = 7;
	 MQ8.ID = 8;
	 MQ135.ID = 135;
	 
	delay_init();	    	 //延时函数初始化	  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
	uart_init(115200);	 //串口初始化为115200
	Adc_Init();
	LED_Init();

 	while(1)
	{
	  //获取MQ3、MQ5、MQ7、MQ135传感器对应气体浓度:乙醇、甲烷、CO、H2
		MQ_printf(&MQ3,MQ3_adc_channel,MQ3_RL,MQ3_R0);
		MQ_printf(&MQ5,MQ5_adc_channel,MQ5_RL,MQ5_R0);
		MQ_printf(&MQ7,MQ7_adc_channel,MQ7_RL,MQ7_R0);
		MQ_printf(&MQ135,MQ135_adc_channel,MQ135_RL,MQ135_R0);
		delay_ms(1000);
	}
 }

最后展示一下我的ONENET上位机:
在这里插入图片描述

总结

本文针对MQ3进行了详细的教程,MQ5、MQ7、MQ135等MQ系列的计算方法基本上都是一样的,如果有任何问题欢迎指正哦!
加入小光嵌入式交流群(qq群号:737327353)免费获取博主所有资料哦!如果群里没有请咨询群主哦!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

UML类图详解

UML类图结构解析 UML类图是一种结构图&#xff0c;用于描述系统的静态结构。它主要用于展示系统中的类&#xff08;class&#xff09;、接口&#xff08;interface&#xff09;、协作&#xff08;collaboration&#xff09;、数据类型&#xff08;data type&#xff09;等以及…

CH341A/B USB转USART/I2C/SPI介绍

CH341A/B USB转USART/I2C/SPI介绍 &#x1f4cd;CH341官方文档&#xff1a;https://www.wch.cn/downloads/CH341DS2_PDF.html CH341A/B是一个USB总线的转接芯片&#xff0c;通过USB总线提供异步串口、打印口、并口以及常用的2线和4线等同步串行接口。 &#x1f341;芯片封装&a…

翻译 《The Old New Thing》 - Returning values from a dialog procedure

Returning values from a dialog procedure - The Old New Thing (microsoft.com)https://devblogs.microsoft.com/oldnewthing/20031107-00/?p41923 Raymond Chen 2003年11月7日 简要 这篇文章由Raymond Chen撰写&#xff0c;解释了对话框过程如何从Windows编程中返回值。他…

如何实现在 Windows 上运行 Linux 程序?

在Windows 上运行Linux程序是可以通过以下几种方法实现: 1.使用 Windows Subsystem for Linux (WSL): WSL是微软提供的功能&#xff0c;可以在Windows 10上运行一个完整的Linux系统。用户可以在Microsoft Store中安装所需的 在开始前我有一些资料&#xff0c;是我根据网友给的…

2024团体程序设计天梯赛L1-103 整数的持续性

题目链接L1-103 整数的持续性 #include<iostream> #include<stdio.h> #include<algorithm> using namespace std; struct node{int x;int d; }p[2000]; bool cmp(node a, node b) {if (a.d b.d) return a.x < b.x;return a.d>b.d; } int cnt, cntt; v…

密钥密码学(三)

原文&#xff1a;annas-archive.org/md5/b5abcf9a07e32fc6f42b907f001224a1 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 第十六章&#xff1a;三次通行协议 本章内容包括 基于指数的三次通行协议 基于矩阵乘法的三次通行协议 基于双边矩阵乘法的三次通行协议 …

UI5 快速入门教程

环境准备 node >16.8 ,VSCode&#xff0c;官方网址 开始 创建一个根文件夹&#xff0c;根文件中创建一个package.json文件 {"name": "quickstart-tutorial","private": true,"version": "1.0.0","author":…

Java基于微信小程序的讲座预约系统的研究与实现,附源码

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…

访问学者申请的成功经验

在申请成为访问学者时&#xff0c;经验是至关重要的。下面知识人网小编将介绍一些可以帮助您成功申请的经验和技巧。 首先&#xff0c;了解目标机构或大学的研究方向和需求是非常重要的。在申请之前&#xff0c;仔细研究该机构的学术项目、研究成果以及教授的专业领域&#xff…

二叉树之AVL树

文章目录 1. AVL树的概念&#xff08;logN)1.1背景1.2规则 2.AVL树节点的定义3.AVL树的插入4. AVL树的旋转(重点&#xff09;4.1 新节点插入较高的右子树的右侧&#xff1a;左单璇&#xff1b;4.2 新节点插入较高左子树的左侧&#xff1a;右单璇&#xff1b;4.3&#xff08;双旋…

文献速递:深度学习胶质瘤诊断---使用深度学习在 MRI 图像中进行低级别胶质瘤的脑肿瘤分割和分级

Title 题目 Brain tumor segmentation and grading of lower-grade glioma using deeplearning in MRI images 使用深度学习在 MRI 图像中进行低级别胶质瘤的脑肿瘤分割和分级 01文献速递介绍 胶质瘤是最常见的脑肿瘤&#xff0c;根据肿瘤的恶性程度和生长速率具有不同的分级…

【高阶数据结构】并查集 -- 详解

一、并查集的原理 1、并查集的本质和概念 &#xff08;1&#xff09;本质 并查集的本质&#xff1a;森林。 &#xff08;2&#xff09;概念 在一些应用问题中&#xff0c;需要将 n 个不同的元素划分成一些不相交的集合。 开始时&#xff0c;每个元素自成一个单元素集合&…

车载诊断的基本框架和概念

车载诊断的基本框架和概念 我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何消耗你的人和事,多看一眼都是你的不对。非必要不费力证明自己,无利益不…

Android开发——ViewPager

适配器 package com.example.myapplication; import android.view.View; import android.view.ViewGroup; import androidx.annotation.AnimatorRes; import androidx.annotation.NonNull; import androidx.viewpager.widget.PagerAdapter; import java.util.ArrayList; publi…

springboot+vue全栈开发【4.前端篇之Vue组件化开发】

目录 前言NPM使用NPM简介nodejs安装npm命令 Vue CLI使用用vue CLI创建一个vue项目 组件化开发组件的构成组件怎么用1.创建一个组件2.在父组件中使用子组件3. 传递数据给子组件4. 监听子组件事件 前言 hi&#xff0c;这个系列是我自学开发的笔记&#xff0c;适合具有一定编程基…

RHCE:网络服务综合项目

基础配置&#xff1a; 1.配置主机名&#xff0c;静态IP地址 2.开启防火墙并配置 3.部分开启SElinux并配置 4.服务器之间使用同ntp.aliyun.com进行时间同步 5.服务器之间实现SSH免密登录 业务需求&#xff1a; 1.Server-NFS-DNS主机配置NFS服务器&#xff0c;将博客网…

Visual Studio 2022 Professional、Enterprise安装教程

Visual Studio 2022 Professional、Enterprise安装教程 下载安装包安装 我是电脑已经有VS2019&#xff0c;现在加装一个VS2022。 下载安装包 首先下载安装包&#xff0c;进入官网进行下载&#xff0c;VS官网下载地址。 进入之后&#xff0c;会显示如下界面&#xff0c;选择Pro…

二、python+前端 实现MinIO分片上传

python前端 实现MinIO分片上传 一、背景二、流程图三、代码 一、背景 问题一&#xff1a;前端 -> 后端 ->对象存储 的上传流程&#xff0c;耗费带宽。 解决方案&#xff1a;上传流程需要转化为 前端 -> 对象存储&#xff0c;节省上传带宽 问题二&#xff1a;如果使用…

近期分享学习心得4

1、带有多的条件的if的语句 逻辑 || 的简写 if (x true || x 2523 || x 小明) {}// 简化操作if ([true, 2523, 小明].includes(x)) {}2、查找两个数组的交集 var numOne [0, 2, 4, 6, 8, 8]; var numTwo [1, 2, 3, 4, 5, 6]; var cross [...new Set(numOne)].filter(item…

《QT实用小工具·三十四》Qt/QML使用WebEngine展示的百度ECharts图表Demo

1、概述 源码放在文章末尾 该项目实现了百度ECharts图表的样式&#xff0c;效果demo如下所示&#xff1a; 项目部分代码如下所示&#xff1a; #include <QGuiApplication> #include <QQmlApplicationEngine> #include <QtWebEngine>int main(int argc, ch…