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)读取传感器输出的模拟电压信号,而我们需要做的则是根据传感器特性曲线函数将电压值转换为一氧化碳浓度值。
模块与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 #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
还没有评论,来说两句吧...