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

认识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)读取传感器输出的模拟电压信号,而我们需要做的则是根据传感器特性曲线函数将电压值转换为一氧化碳浓度值。

MQ-7的传感器特性曲线

模块与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的引脚

MQ-7与STM32F407ze的连接

代码功能实现

        代码上的大致思路为:定义模块引脚和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 = 42MHZADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;	//禁止MDAADC_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);// 使能 ADC2ADC_Cmd(ADC2, ENABLE);// 初始化 ADC2ADC_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发送数据到PCUSART_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 = 21MHzSysTick_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;
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

minio客户端基本操作

minio客户端基本操作 桶 创建桶 如果要创建新的桶 输入名称&#xff0c;点击创建即可&#xff0c;默认权限就行 删除桶 点击要删除的桶 点击删除 修改桶 如果哪天需要修改桶的权限或者其他信息&#xff0c;还是先点击这个桶进入详情 然后点击要修改的属性&#xff0c;选择…

Qt5.15.2静态编译 VS2017 with static OpenSSL

几年前编译过一次Qt静态库:VS2015编译Qt5.7.0生成支持XP的静态库,再次编译,毫无压力。 一.环境 系统:Windows 10 专业版 64位 编译器:visual studio 2017 第三方工具:perl,ruby和python python用最新的3.x.x版本也是可以的 这三个工具都需要添加到环境变量,安装时勾选…

JavaScript 的初步学习上篇

JavaScript 的介绍 JavaScript 之父 布兰登 . 艾奇 (Brendan Eich) ,1995 年, 用 10 天时间完成 JavaScript 的设计. JavaScript 和 Java 的关系 两者之间就像老婆和老婆饼的关系,即毫无关系, JavaScript 最初的名字叫LiveScript,为了蹭 Java 热度,才改名为 JavaScript.JavaScr…

【2023 云栖】阿里云田奇铣:大模型驱动 DataWorks 数据开发治理平台智能化升级

云布道师 本文根据 2023 云栖大会演讲实录整理而成&#xff0c;演讲信息如下&#xff1a; 演讲人&#xff1a;田奇铣 | 阿里云 DataWorks 产品负责人 演讲主题&#xff1a;大模型驱动 DataWorks 数据开发治理平台智能化升级 随着大模型掀起 AI 技术革新浪潮&#xff0c;大数…

C#中openFileDialog控件的使用方法

目录 一、OpenFileDialog基本属性 二、使用 OpenFile 从筛选的选择中打开文件 1.示例源码 2.生成效果 3. 其它示例 三、使用 StreamReader 以流的形式读取文件 1.示例源码 2.生成效果 四、一种新颖的Windows窗体应用文件设计方法 在C#中&#xff0c;OpenFileDialog控件…

AIGC,ChatGPT AI绘画 Midjourney 注册流程详细步骤

AI 绘画,Midjourney完成高清图片绘制,轻松掌握AI工具。 前期准备: ① 一个能使用的谷歌账号 ② 可以访问外网 Midjourney注册 1.进入midjourney官网https://www.midjourney.com 点击左下角”Join the Beta”,就可以注册,第一次使用的小伙伴会弹出提示,只需要点击Acc…

C语言程序设计知识点总结归纳(全书)

C知识点总结归纳目录 第一章 程序设计和C语言一、C的入门小概念二、程序设计的问题三、首先要搞清楚编译器、编辑器和IDE的区别 第二章 算法——程序的灵魂一、程序算法数据结构二、算法的特性三、怎样表示一个算法四、结构化程序的设计方法 第三章 C程序设计——顺序程序设计一…

vue3+ts 依赖注入 provide inject

父级&#xff1a; <template><div><h1>App.vue (爷爷级别)</h1><label><input type"radio" v-model"colorVal" value"red" name"color" />红色</label><label><input type"r…

【STM32】GPIO输出

1 GPIO简介 &#xff08;1&#xff09;GPIO&#xff08;General Purpose Input Output&#xff09;通用输入输出口 &#xff08;2&#xff09;可配置为8种输入输出模式 &#xff08;3&#xff09;引脚电平&#xff1a;0V~3.3V&#xff0c;部分引脚可容忍5V&#xff08;可以输…

因式分解的几何意义

本来准备和女儿一起玩一道几何题&#xff0c;想想还是算了&#xff0c;不如讲点更有趣的。 任何因式分解都是在堆积木&#xff0c;不信你看&#xff1a; 二项式定理&#xff0c;洋灰三角&#xff0c;都是面积&#xff0c;体积&#xff0c;超维体积的拼接&#xff0c;一个大超…

Linux学习笔记之六(进程之间的管道通信和信号处理)

目录 1、管道通信1.1、无名管道1.1、有名管道 2、信号处理2.1、信号的种类和发送2.2、信号的接受和处理 1、管道通信 管道通信是一个设备中进程与进程之间通信的一种方式&#xff0c;分为无名管道和有名管道两种。前者只能用于有亲缘关系的进程之间的通信&#xff0c;如父子进…

RC-MVSNet:无监督的多视角立体视觉与神经渲染--论文笔记(2022年)

RC-MVSNet&#xff1a;无监督的多视角立体视觉与神经渲染--论文笔记&#xff08;2022年&#xff09; 摘要1 引言2 相关工作2.1 基于监督的MVS2.2 无监督和自监督MVS2.3 多视图神经渲染 3 实现方法3.1 无监督的MVS网络 Chang, D. et al. (2022). RC-MVSNet: Unsupervised Multi-…

leetcode面试经典150题——32 串联所有单词的子串(中等+困难)

题目&#xff1a; 串联所有单词的子串(1中等) 描述&#xff1a; 给定两个字符串 s 和 p&#xff0c;找到 s 中所有 p 的 异位词 的子串&#xff0c;返回这些子串的起始索引。不考虑答案输出的顺序。 异位词 指由相同字母重排列形成的字符串&#xff08;包括相同的字符串&…

Linux 命令vim(编辑器)

(一)vim编辑器的介绍 vim是文件编辑器&#xff0c;是vi的升级版本&#xff0c;兼容vi的所有指令&#xff0c;同时做了优化和延伸。vim有多种模式&#xff0c;其中常用的模式有命令模式、插入模式、末行模式&#xff1a;。 (二)vim编辑器基本操作 1 进入vim编辑文件 1 vim …

百度智能云正式上线Python SDK版本并全面开源

文章目录 前言一、SDK的优势二、千帆SDK&#xff1a;快速落地LLM应用三、如何快速上手千帆SDK3.1、SDK快速启动3.2. SDK进阶指引 3.3. 通过Langchain接入千帆SDK4、开源社区 前言 百度智能云千帆大模型平台再次升级&#xff01;在原有API基础上&#xff0c;百度智能云正式上线…

单链表原来是这样实现的!

文章目录 前言1. 链表的概念及结构1.1在链表里&#xff0c;每节“车厢”是什么样的呢&#xff1f;1.2为什么还需要指针变量来保存下⼀个节点的位置&#xff1f; 2. 单链表的实现1. 定义结构体(Seqlist)2. 打印函数(SLTPrint)小插曲&#xff0c;创建节点函数CreateNode3. 尾插函…

Qt 串口编程-从入门到实战

1. Qt 串口通信流程解析 1.1 串行通信和并行通信对比 并行通信适合距离较短的通信&#xff0c;且信号容易受干扰&#xff0c;成本高串口通讯-设备&#xff08;蓝牙&#xff0c; wifi&#xff0c; gprs&#xff0c; gps&#xff09; 1.2 Qt 串口通信具体流程 1. 创建 QSerial…

学习Pandas 二(Pandas缺失值处理、数据离散化、合并、交叉表与透视表、分组与聚合)

文章目录 六、高级处理-缺失值处理6.1 检查是否有缺失值6.2 缺失值处理6.3 不是缺失值NaN&#xff0c;有默认标记的 七、高级处理-数据离散化7.1 什么是数据的离散化7.2 为什么要离散化7.3 如何实现数据的离散化 八、高级处理-合并8.1 pc.concat实现合并&#xff0c;按方向进行…

面试常见问题:什么是进程? 什么是线程?进程和线程有什么区别?

1.什么是进程&#xff1f; 进程是操作系统中一个程序在执行过程中的一个实例&#xff0c;每个进程都有自己独立的地址空间&#xff0c;进程间不共享内存。它是程序运行的最小内存单元&#xff1b; 进程特点&#xff1a; 1> 需要占用独立的内存空间&#xff1b; 2>可以并…

2023年最新PyCharm安装详细教程及pycharm配置

目录 一、PyCharm简介及其下载网站 二、单击网站的Downloads&#xff0c;进入二级页面&#xff0c;选择对应的操作系统下载PyCharm 三、PyCharm的安装程序的安装及其配置(configuration) 1、运行PyCharm Setup 2、安装位置设置 3、安装选项设置 4、开始菜单中PyCharm快捷方式的…