草庐IT

MQ-7一氧化碳传感器模块功能实现(STM32)

nazonomaster 2023-04-03 原文

认识MQ-7模块与其工作原理

        首先来认识MQ-7模块,MQ-7可以检测空气中的一氧化碳(CO)浓度。他采用半导体气敏元件来检测CO的气体浓度,其灵敏度高、反应速度快、响应时间短、成本低廉等特点使得它被广泛应用于智能家居、工业自动化、环保检测等领域。

        MQ-7模块的工作原理基于半导体材料在气体作用下发生电学性质的变化,通过测量这种变化来检测气体浓度。具体来说,MQ-7烟雾传感器内部有一块小小的加热器,通过电流加热,使得传感器的工作环境温度升高,同时加速气体分子的运动。模块中的半导体气敏元件表面涂有一层特殊的金属氧化物,当在特定温度下,空气中的CO气体与金属氧化物接触时,会发生氧化还原反应,使得金属氧化物表面的电阻发生变化,当空气中的CO浓度增加时,电阻减小,从而输出一个模拟信号。 

        回到MQ-7模块上,MQ-7一共有4个引脚,分别是GND、DOUT、AOUT和VCC。VCC是电源引脚,用于给模块供电,可以理解为正极,一般接3.3V或者5V。GND连接模块的地线,可以理解为是负极,通常是0V。DOUT是数字输出引脚,用于输出数字信号。AOUT是模拟输出引脚,用于输出模拟信号。我们一般使用AOUT引脚也就是读取模块输出的模拟信号。

        32单片机则是可以通过模数转换器(ADC)读取传感器输出的模拟电压信号,而我们需要做的则是根据传感器特性曲线函数将电压值转换为一氧化碳浓度值。

模块与32单片机的连接

        这里我使用的单片机型号为STM32F407ze,上面也说过,要获得MQ-7测量的数据就要使用模数转换器(ADC),所以需要选择具有ADC功能的引脚,哪一个引脚拥有ADC功能可以通过查阅单片机的资料可知。

        通过查阅资料,我选择具有ADC功能的PA6引脚,PA6与CAMERA的第16引脚相连,所以只需要将MQ-7的AOUT引脚通过杜邦线接到CAMERA的第16引脚即可。剩下的就是接电源线了,因为CAMERA的第1引脚可以输出3.3V的电压,第2引脚接地,所以只需要将模块的VCC与GND与他们相连即可。

代码功能实现

        代码上的大致思路为:定义模块引脚和ADC后,首先获取ADC在一定时间内的平均值,其次,可以利用公式将ADC的平均值转化为MQ-7的输出电压值,接着根据这个电压信号和传感器特性曲线计算出CO的浓度。 

ADC部分

       adc.c

        因为要使用ADC功能,所以此段代码第一个函数用于定义使用ADC的引脚和功能,第二个函数用于获取ADC的值。

#include "adc.h"

void adc_Init()
{
    GPIO_InitTypeDef GPIO_InitStructure;
	ADC_CommonInitTypeDef ADC_CommonInitStructure;
    ADC_InitTypeDef ADC_InitStructure;

    // 使能 ADC 引脚的 GPIO 时钟
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);

    // 将 ADC 引脚配置为模拟输入
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    // 使能 ADC2 时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC2, ENABLE);
	
	/* ADC的常规配置 */
	ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;		//独立模式
	ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div2;		//84MHZ/2 = 42MHZ
	ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;	//禁止MDA
	ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles;	//ADC通道采用间隔
	ADC_CommonInit(&ADC_CommonInitStructure);

    // ADC初始化
    ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;//分辨率
    ADC_InitStructure.ADC_ScanConvMode = DISABLE;//禁止扫描	
    ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;//连续转换模式
    ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;//不需要外部触发
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//数据右对齐
    ADC_InitStructure.ADC_NbrOfConversion = 1;//一次转换
    ADC_Init(ADC2, &ADC_InitStructure);

    //ADC2的采样通道6放入规则通道1中
    ADC_RegularChannelConfig(ADC2, ADC_Channel_6, 1, ADC_SampleTime_3Cycles);

    // 使能 ADC2
    ADC_Cmd(ADC2, ENABLE);

    // 初始化 ADC2
    ADC_SoftwareStartConv(ADC2);
}


//获得 ADC 值
uint16_t ADC_Read(void)
{
    // 开始转换
    ADC_SoftwareStartConv(ADC2);

    // 等到转换完成
    while (ADC_GetFlagStatus(ADC2, ADC_FLAG_EOC) == RESET);

    // 获取转换结果
    return ADC_GetConversionValue(ADC2);
}

        adc.h 

#ifndef _ADC_H
#define _ADC_H

//C文件中需要的其他的头文件
#include <stm32f4xx.h>
#include "sys.h"
#include "delay.h"


//C文件中定义的函数的声明
void adc_Init(void);
uint16_t ADC_Read(void);

#endif

MQ-7部分

        MQ7.c

#include "mq7.h"

#define CAL_PPM  10  // 校准环境中PPM值
#define RL	     10  // RL阻值
#define R0	     16  // R0阻值

u16 MQ7_Buffer[4];

void CO_Init(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);//使能PA的时钟
	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;			//PA6 模拟输入引脚
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;		//输入模式
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;	//上下拉电阻:无上下拉电阻
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	adc_Init();
}

//计算平均值
u16 ADC2_Average_Data(u8 ADC_Channel)
{
	u16 temp_val=0;
	u8 t;
	for(t=0;t<CO_READ_TIMES;t++)	//#define CO_READ_TIMES	10	定义烟雾传感器读取次数,读这么多次,然后取平均值

	{
		temp_val+=ADC_Read();	//读取ADC值
		delay_ms(5);
	}
	temp_val/=CO_READ_TIMES;//得到平均值
    return (u16)temp_val;//返回算出的ADC平均值
}

//float voltage = adc_value * (3.3 / 4096.0);  将ADC值转换为电压

/***********************************
		计算Rs的两种公式
float Rs = (3.3 * RL) / voltage - RL; 计算传感器的电阻  RL:负载电阻阻值
float RS = (3.3f - voltage) / voltage * RL;
************************************/

//float co_ppm = a * pow(Rs/R0, b); 使用校准曲线计算一氧化碳浓度
//a, b是MQ-7传感器模块校准曲线的系数.其值来源于MQ7的手册资料,a = 98.322, b = -1.458
//R0是器件在洁净空气中的电阻值,来自于MQ-7灵敏度特性曲线,R0 = RS / pow(CAL_PPM / 98.322, 1 / -1.458f);

//读取MQ7传感器的电压值
float CO_Get_Vol()
{
	u16 adc_value = 0;//这是从MQ-7传感器模块电压输出的ADC转换中获得的原始数字值,该值的范围为0到4095,将模拟电压表示为数字值
	float voltage = 0;//MQ-7传感器模块的电压输出,与一氧化碳的浓度成正比
	
	adc_value = ADC2_Average_Data(ADC_Channel_6);
	delay_ms(5);
	
    voltage  = (3.3/4096.0)*(adc_value);
	
	return voltage;
}

/*********************
// 传感器校准函数,根据当前环境PPM值与测得的RS电压值,反推出R0值。
// 经过个人测试,在空气中测出R0为16
float MQ7_PPM_Calibration()
{
	float RS = 0;
	float R0 = 0;
	RS = (3.3f - Smog_Get_Vol()) / Smog_Get_Vol() * RL;//RL	10  // RL阻值
	R0 = RS / pow(CAL_PPM / 98.322, 1 / -1.458f);//CAL_PPM  10  // 校准环境中PPM值
	return R0;
}
**********************/


// 根据公式计算CO_ppm
float MQ7_GetPPM()
{
	float RS = (3.3f - CO_Get_Vol()) / CO_Get_Vol() * RL;
	float ppm = 98.322f * pow(RS/R0, -1.458f);
	return  ppm;
}

        mq7.h

#ifndef _MQ7_H
#define _MQ7_H

//C文件中需要的其他的头文件
#include <stm32f4xx.h>
#include "sys.h"
#include "delay.h"
#include "math.h"
#include "adc.h"

#define SMOG_PIN46_R	1000			//烟雾传感器管脚4、6接出到地的电阻值
#define CO_READ_TIMES	10				//定义CO传感器读取次数,读这么多次,然后取平均值

//C文件中定义的函数的声明
void CO_Init(void);
u16 ADC2_Average_Data(u8 ADC_Channel);
float CO_Get_Vol(void);
//float MQ7_PPM_Calibration();
float MQ7_GetPPM(void);

#endif

主函数部分

        在主函数中调用 MQ7_GetPPM() 函数即可。

#include <stm32f4xx.h>
#include "sys.h"
#include <stdio.h>
#include "delay.h"
#include "uart.h"
#include "mq7.h"

//重定向fputc函数
int fputc(int ch, FILE *F)
{
	//通过串口1发送数据到PC
	USART_SendData(USART1, ch);
	while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);//等待发送数据完毕
	
	return ch;
}

u8 Smog_yu = 30;//Smog的阈值	

int main(void)
{
	float CO_ppm = 0;
	
	//确定系统定时器的工作频率  内核的工作频率/8 = 168MHz/8 = 21MHz
	SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
	
	UART1_Config();//配置串口
	
	CO_Init();//配置CO模块
	
	while(1)
	{
		CO_ppm = MQ7_GetPPM();
		printf("CO_ppm:%.3f    CO阈值:%d\n",CO_ppm, CO_yu);
			
	}
	
	return 0;
}

有关MQ-7一氧化碳传感器模块功能实现(STM32)的更多相关文章

  1. STM32读取串口传感器数据(颗粒物传感器,主动上传) - 2

    文章目录1.开发板选择*用到的资源2.串口通信(个人理解)3.代码分析(注释比较详细)1.主函数2.串口1配置3.串口2配置以及中断函数4.注意问题5.源码链接1.开发板选择我用的是STM32F103RCT6的板子,不过代码大概在F103系列的板子上都可以运行,我试过在野火103的霸道板上也可以,主要看一下串口对应的引脚一不一样就行了,不一样的就更改一下。*用到的资源keil5软件这里用到了两个串口资源,采集数据一个,串口通信一个,板子对应引脚如下:串口1,TX:PA9,RX:PA10串口2,TX:PA2,RX:PA32.串口通信(个人理解)我就从串口采集传感器数据这个过程说一下我自己的理解,

  2. STM32的HAL和LL库区别和性能对比 - 2

    LL库和HAL库简介LL:Low-Layer,底层库HAL:HardwareAbstractionLayer,硬件抽象层库LL库和hal库对比,很精简,这实际上是一个精简的库。LL库的配置选择如下:在STM32CUBEMX中,点击菜单的“ProjectManager”–>“AdvancedSettings”,在下面的界面中选择“AdvancedSettings”,然后在每个模块后面选择使用的库总结:1、如果使用的MCU是小容量的,那么STM32CubeLL将是最佳选择;2、如果结合可移植性和优化,使用STM32CubeHAL并使用特定的优化实现替换一些调用,可保持最大的可移植性。另外HAL和L

  3. ESP32学习入门:WiFi连接网络 - 2

    目录一、ESP32简单介绍二、ESP32Wi-Fi模块介绍三、ESP32Wi-Fi编程模型四、ESP32Wi-Fi事件处理流程 五、ESP32Wi-Fi开发环境六、ESP32Wi-Fi具体代码七、ESP32Wi-Fi代码解读6.1主程序app_main7.2自定义代码wifi_init_sta()八、ESP32Wi-Fi连接验证8.1测试方法8.2服务器模拟工具sscom58.3测试代码8.4测试结果前言为了开发一款亚马逊物联网产品,开始入手ESP32模块。为了能够记录自己的学习过程,特记录如下操作过程。一、ESP32简单介绍ESP32是一套Wi-Fi(2.4GHz)和蓝牙(4.2)双模解决方

  4. Spring Security 6.0系列【32】授权服务器篇之默认过滤器 - 2

    有道无术,术尚可求,有术无道,止于术。本系列SpringBoot版本3.0.4本系列SpringSecurity版本6.0.2本系列SpringAuthorizationServer版本1.0.2源码地址:https://gitee.com/pearl-organization/study-spring-security-demo文章目录前言1.OAuth2AuthorizationServerMetadataEndpointFilter2.OAuth2AuthorizationEndpointFilter3.OidcProviderConfigurationEndpointFilter4.N

  5. ruby - 摘要::CRC32 与 Zlib - 2

    在我的代码中,我需要使用各种算法(包括CRC32)对文件进行哈希处理。因为我还在Digest系列中使用其他加密哈希函数,所以我认为为它们维护一个一致的接口(interface)会很好。为了记录,我确实找到了digest-crc,一颗完全符合我要求的gem。问题是,Zlib是标准库的一部分,并且有一个我想重用的CRC32工作实现。此外,它是用C编写的,因此它应该提供与digest-crc相关的卓越性能,后者是纯ruby​​实现。实现Digest::CRC32一开始看起来非常简单:%w(digestzlib).each{|f|requiref}classDigest::CRC32一切正常:

  6. ruby - 安装gem : Couldn't reserve space for cygwin's heap, Win32错误487错误 - 2

    我正在尝试在我的机器上安装win32-apigem,但在构建native扩展时我遇到了一些问题:$geminstallwin32-api--no-ri--rdocTemporarilyenhancingPATHtoincludeDevKit...Buildingnativeextensions.Thiscouldtakeawhile...C:\Programs\dev_kit\bin\make.exe:***Couldn'treservespaceforcygwin'sheap,Win32error0ERROR:Errorinstallingwin32-api:ERROR:Failed

  7. Ruby 1.9 - 没有这样的文件可以加载 'win32/open3' - 2

    我在Windows上运行ruby​​1.9.2并试图移植在Ruby1.8中工作的代码。该代码使用以前运行良好的Open4.popen4。对于1.9.2,我做了以下事情:通过geminstallPOpen4安装了POpen4需要POpen4通过require'popen4'尝试像这样使用POpen4:Open4.popen4("cmd"){|io_in,io_out,io_er|...}当我这样做时,我得到了错误:nosuchfiletoload--win32/open3如果我尝试安装win32-open3(geminstallwin32-open3),我会收到错误消息:win32-op

  8. Dell Inspiron 5488加内存32G - 2

    DellInspiron5488加内存32G 原装内置内存仅仅8G,目前看,真的太小了! 1.内存型号Dell5488内存型号:DDR42666。笔记本有两个内存插槽,原装占了一个,还能扩展一个。 2.买内存如果买Dell原装笔记本内存,8G就得500块左右。 我咨询了一下,三星的笔记本内存,可以兼容。16G,299块(2023年2月23日,京东价) Dell5488内存组合,最多只能插两根16G内存。 我于是买了两根三星16G内存。装上,很爽😄 跑国产系统统信UOS,再也看不到用交换区了,32G内存,爽!  

  9. 蓝桥杯 stm32 MCP4017 - 2

    本文代码使用HAL库。文章目录前言一、MCP4017的重要特性二、MCP4017计算RBW阻值三、MCP4017地址四、MCP4017读写函数五、CubeMX创建工程(利用ADC测量MCP4017电压)、对应代码:总结前言一、MCP4017的重要特性蓝桥杯板子上的是MCP4017T-104ELT,如图1。MCP4017是一个可编程电阻,通过写入的数值可以改变电阻的大小。重点在于6引脚(W),5引脚(B&#

  10. STM32 OTA应用开发——通过USB实现OTA升级 - 2

    STM32OTA应用开发——通过USB实现OTA升级目录STM32OTA应用开发——通过USB实现OTA升级前言1环境搭建2功能描述3BootLoader的制作4APP的制作5烧录下载配置6运行测试结束语前言什么是OTA?百度百科:空中下载技术(Over-the-AirTechnology;OTA),是通过移动通信的空中接口实现对移动终端设备及SIM卡数据进行远程管理的技术。经过公网多年的应用与发展,已十分成熟,网络运营商通过OTA技术实现SIM卡远程管理,还能提供移动化的新业务下载功能。实际上,现在我们所说的OTA比百度百科的定义还要更广泛,OTA的形式已经不再局限于手机和SIM卡,只要涉及

随机推荐