1.演示视频
基于stm32单片机家庭环境监测系统视频演示_哔哩哔哩_bilibili
2. 项目介绍
本项目基于STM32F103C8T6最小系统板,打造了一个功能强大、显示直观的家庭环境监测系统。系统集成了温湿度、光照强度、多种气体、人体感应等传感器,并通过OLED屏幕本地实时显示数据,同时支持手机蓝牙小程序远程监控与告警。它不仅是一个全面的环境数据采集站,更是一个具备布防/撤防功能的简易家庭安防系统。
3. 系统功能亮点
- 环境多参数监测:实时采集温度、湿度、光照强度、空气质量(CO/CO2/烟雾等综合指标)。
- 人体红外感应:检测指定区域内是否有人体活动。
- 双模远程通信:支持ESP32 Wi-Fi模块(预留扩展接口)和HC-05/06蓝牙模块与手机连接。
- 手机小程序监控:通过自开发的微信小程序,可远程查看所有传感器数据,并进行系统控制。
- 智能安防布防:具备"布防"和"撤防"两种模式。在布防模式下,检测到人体入侵将触发本地蜂鸣器高声警报并推送手机提醒。
- 交互与提示:配备四个功能按键和一个蜂鸣器,方便本地操作和声音提示。
4.硬件设计
4.1 硬件组成 (BOM List)
模块名称 | 型号/规格 | 通信协议 | 功能 |
---|---|---|---|
主控芯片 | STM32F103C8T6 (最小系统板) | - | 系统核心,负责数据处理与逻辑控制 |
显示模块 | 0.96英寸 OLED屏 | I2C | 本地实时显示所有传感器数据及系统状态 |
温湿度传感器 | AHT20 | I2C | 高精度测量环境温度和湿度 |
光照传感器 | GY-30 (BH1750) | I2C | 检测环境光照强度 (Lux) |
气体传感器 | MQ-135 | 模拟/数字 | 综合检测空气质量、CO2、CO、烟雾、苯等有害气体 |
人体红外传感器 | HC-SR501 | 数字GPIO | 检测人体移动 |
蓝牙模块 | HC-05 或 HC-06 | UART | 与手机小程序进行串口蓝牙通信 |
Wi-Fi模块 | ESP-01S (ESP8266) 或 ESP32 | UART | 预留物联网扩展接口,可接入云平台 |
声学提示 | 有源蜂鸣器 | GPIO | 发出警报和提示音 |
输入控制 | 4x轻触按键 | GPIO | 本地切换模式、设置参数等 |
供电接口 | Type-C | - | 5V供电,并配有电源开关 |
4.2主要芯片介绍
STM32F103C8T6介绍
STM32F103C8T6是一款由意法半导体公司(ST)推出的基于Cortex-M3内核的32位微控制器,硬件采用LQFP48封装,属于ST公司微控制器中的STM32系列。
AHT20温湿度传感器介绍
AHT20是一款高精度数字温湿度传感器,采用I2C通信(地址0x38),供电需3.3V。其温度精度±0.3℃,湿度精度±2%RH,出厂已校准,即插即用。相比DHT11,精度更高、通信更稳定,是环境监测项目的理想选择。
BH1750光照强度传感器介绍
BH1750FVI 是一款数字式环境光强度传感器。它通过 I2C 接口直接输出光照度值(单位:Lux),测量范围 1 - 65535 lx。具有高精度、无需外部元件等优点,是替代光敏电阻的理想选择,广泛应用于自动背光调节和智能照明系统。
ECB01H2S蓝牙模块介绍
ECB01H2S 是一款基于蓝牙5.0的串口透传模块 。它将复杂的蓝牙协议封装成简单的串口通信,用户只需通过AT指令进行配置,即可让主控设备(如STM32)通过串口与手机等蓝牙主机实现无线数据双向传输,极大简化了无线通信功能的开发流程,广泛应用于智能硬件和数据传输项目。
I2C OLED显示屏介绍
该屏幕拥有128x64 高分辨率,采用I2C接口,仅需两根信号线即可驱动。其像素点自发光,显示黑色时功耗极低,无需背光,视觉效果清晰锐利,对比度高,是嵌入式项目中最受欢迎的小型显示方案。
ESP8266WiFi模块介绍
ESP8266 是一款低成本、高性能的Wi-Fi SOC芯片模块。其核心功能是通过串口AT指令与单片机通信,轻松实现设备联网。它内置了TCP/IP协议栈,可作为独立MCU或Wi-Fi适配器使用,是物联网项目中实现无线连接最经典、性价比最高的解决方案之一。
5.系统设计与实现
5.1整体系统设计
5.2 数据流与工作逻辑
- 数据采集层:STM32主控循环读取各个传感器的数据。
- 数据处理层:对原始数据进行滤波、校准和计算,得到有意义的物理量(如℃、%RH、Lux、空气质量等级)。
- 本地显示层:处理后的数据通过I2C协议发送到OLED屏幕进行刷新显示。
- 远程通信层:STM32通过串口将数据按照自定义协议格式发送给蓝牙模块,再由蓝牙传输至手机小程序。手机下发的控制指令(如布防)也通过此路径逆向传输给STM32。
逻辑控制层 :主控根据当前系统模式(布防/撤防)和传感器状态(是否检测到人)执行相应的动作,如控制蜂鸣器。
- 本地显示效果:
屏幕第一页显示:温度: 26.5℃
, 湿度: 45%
, 光照: 256 Lux
。
屏幕循环显示气体数据:Air Q: Good
, Gas: None
, PM2.5: 15
, Smoke: Low
。
人体检测状态:P: Yes
(有人) 或 P: No
(无人)。
安防状态:Security On
(布防) 或 Security Off
(撤防)。
- 手机监控:
手机小程序成功连接蓝牙后,界面同步显示所有环境数据。
设有"布防/撤防"切换按钮。
在布防状态下,若检测到人体,手机端会弹出弹窗警报,并显示"警戒中!"。
- 安防联动:
撤防模式 :检测到人体,仅在屏幕和手机端显示"有人",无声音警报。(适合家中有人时)
布防模式 :检测到人体,蜂鸣器立即鸣叫 ,同时手机端弹出警报。(适合出门或夜间)
6.代码展示
/* USER CODE BEGIN Header */
/**
******************************************************************************
* @file : main.c
* @brief : Main program body - 简化版
******************************************************************************
*/
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "oled.h"
#include "temp_humi.h"
#include "gas_sensor.h"
#include "light_sensor.h"
#include "bluetooth.h"
#include "gas_display.h"
#include "buzzer.h"
#include <stdio.h>
#include <string.h>
/* Private variables ---------------------------------------------------------*/
I2C_HandleTypeDef hi2c1;
ADC_HandleTypeDef hadc1;
UART_HandleTypeDef huart1;
/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
void MX_GPIO_Init(void);
void MX_I2C1_Init(void);
void MX_ADC1_Init(void);
void MX_USART1_UART_Init(void);
/**
* @brief IR传感器初始化 - 简化版
* @param None
* @retval None
*/
void IR_Sensor_Init(void)
{
GPIO_InitTypeDef GPIO_InitStruct = {0};
__HAL_RCC_GPIOB_CLK_ENABLE();
GPIO_InitStruct.Pin = GPIO_PIN_9;
GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
GPIO_InitStruct.Pull = GPIO_PULLUP;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
}
/**
* @brief 检测人体存在 - 简化版
* @param None
* @retval 1: 检测到人体, 0: 未检测到人体
*/
uint8_t IR_Sensor_Detect(void)
{
return (HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_9) == GPIO_PIN_RESET) ? 1 : 0;
}
/**
* @brief 应用程序入口点
* @retval int
*/
int main(void)
{
/* 初始化HAL库 */
HAL_Init();
/* 配置系统时钟 */
SystemClock_Config();
/* 初始化所有外设 */
MX_GPIO_Init();
MX_I2C1_Init();
MX_ADC1_Init();
MX_USART1_UART_Init();
/* 初始化所有传感器和外设 */
OLED_Init();
TEMP_HUMI_Init();
GAS_SENSOR_Init();
LIGHT_SENSOR_Init();
IR_Sensor_Init();
Bluetooth_Init();
BUZZER_Init();
/* 显示初始屏幕 */
OLED_Clear();
OLED_ShowString(0, 0, "Starting...", 12);
OLED_Refresh();
/* 初始化变量 */
float temperature = 0.0f;
float humidity = 0.0f;
float light = 0.0f;
uint8_t ir_status = 0;
uint8_t last_ir_status = 0;
uint8_t bt_send_counter = 0;
/* 创建气体数据结构 */
GAS_DATA gas_data = {0};
/* 气体传感器初始校准 */
GAS_SENSOR_Calibrate(400.0f);
/* 手动设置一些初始值,确保数据不为空 */
gas_data.raw_ppm = 500.0f;
gas_data.co2_equivalent = 400.0f;
gas_data.combustible_gas = 150.0f;
gas_data.smoke_density = 75.0f;
gas_data.pm25 = 50.0f;
gas_data.aqi = 100;
gas_data.change_rate = 0.5f;
gas_data.gas_type = GAS_TYPE_NORMAL;
gas_data.safety_level = SAFETY_LEVEL_SAFE;
gas_data.leak_detected = 0;
gas_data.fire_risk = 2;
/* 无限循环 */
while (1)
{
/* 检查布防/撤防状态是否改变 */
if (Bluetooth_GetAndClearStatusChanged()) {
uint8_t armed_status = Bluetooth_GetArmedStatus();
DISPLAY_ArmedStatus(armed_status, 3000);
if (armed_status) {
BUZZER_LoudAlarm();
} else {
BUZZER_Beep(100);
}
}
/* 读取传感器数据 */
TEMP_HUMI_ReadData(&temperature, &humidity);
if (LIGHT_SENSOR_IsConnected()) {
LIGHT_SENSOR_ReadLight(&light);
}
ir_status = IR_Sensor_Detect();
/* 更新气体传感器数据 - 确保每次循环都更新 */
uint16_t adc_value = GAS_SENSOR_Read();
if (adc_value > 0) {
GAS_SENSOR_UpdateData(&gas_data);
}
/* 人体检测处理 */
if (ir_status && !last_ir_status) {
/* 更新显示 */
DISPLAY_UpdateEnvData(temperature, humidity, light, ir_status);
DISPLAY_GasData(&gas_data);
/* 发送数据到蓝牙 - 恢复原来的两次发送格式 */
/* 第一次发送基本环境数据 */
char bt_data1[150];
sprintf(bt_data1, "{\"temp\":%.1f,\"humi\":%.1f,\"light\":%.1f,\"human\":%d,\"aqi\":%d,\"pm25\":%.1f}\r\n",
temperature, humidity, light, ir_status, gas_data.aqi, gas_data.pm25);
Bluetooth_SendString(bt_data1);
/* 短暂延时,确保第一条数据发送完成 */
HAL_Delay(20);
/* 第二次发送气体详细数据 */
char bt_data2[150];
sprintf(bt_data2, "{\"co2\":%.1f,\"ppm\":%.1f,\"gas_type\":%d,\"safety\":%d,\"leak\":%d,\"fire\":%d,\"rate\":%.2f}\r\n",
gas_data.co2_equivalent, gas_data.raw_ppm, gas_data.gas_type,
gas_data.safety_level, gas_data.leak_detected, gas_data.fire_risk, gas_data.change_rate);
Bluetooth_SendString(bt_data2);
/* 布防状态下发出警报 */
if (Bluetooth_GetArmedStatus()) {
BUZZER_EmergencyAlarm();
}
}
last_ir_status = ir_status;
/* 更新显示 */
DISPLAY_UpdateEnvData(temperature, humidity, light, ir_status);
DISPLAY_GasData(&gas_data);
/* 处理蓝牙数据 */
Bluetooth_Process();
/* 每5秒通过蓝牙发送环境数据 */
if (++bt_send_counter >= 5) {
bt_send_counter = 0;
/* 恢复原来的两次发送格式 */
/* 第一次发送基本环境数据 */
char bt_data1[150];
sprintf(bt_data1, "{\"temp\":%.1f,\"humi\":%.1f,\"light\":%.1f,\"human\":%d,\"aqi\":%d,\"pm25\":%.1f}\r\n",
temperature, humidity, light, ir_status, gas_data.aqi, gas_data.pm25);
Bluetooth_SendString(bt_data1);
/* 短暂延时,确保第一条数据发送完成 */
HAL_Delay(50);
/* 第二次发送气体详细数据 */
char bt_data2[150];
sprintf(bt_data2, "{\"co2\":%.1f,\"ppm\":%.1f,\"gas_type\":%d,\"safety\":%d,\"leak\":%d,\"fire\":%d,\"rate\":%.2f}\r\n",
gas_data.co2_equivalent, gas_data.raw_ppm, gas_data.gas_type,
gas_data.safety_level, gas_data.leak_detected, gas_data.fire_risk, gas_data.change_rate);
Bluetooth_SendString(bt_data2);
}
/* 切换LED指示灯状态 */
HAL_GPIO_TogglePin(GPIOC, GPIO_PIN_13);
/* 延时1秒 */
HAL_Delay(1000);
}
}
/**
* @brief 系统时钟配置
* @retval None
*/
void SystemClock_Config(void)
{
RCC_OscInitTypeDef RCC_OscInitStruct = {0};
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
RCC_PeriphCLKInitTypeDef PeriphClkInit = {0};
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
RCC_OscInitStruct.HSEState = RCC_HSE_ON;
RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
RCC_OscInitStruct.HSIState = RCC_HSI_ON;
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
HAL_RCC_OscConfig(&RCC_OscInitStruct);
RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2);
/* 配置ADC时钟 */
PeriphClkInit.PeriphClockSelection = RCC_PERIPHCLK_ADC;
PeriphClkInit.AdcClockSelection = RCC_ADCPCLK2_DIV6;
HAL_RCCEx_PeriphCLKConfig(&PeriphClkInit);
}
/**
* @brief GPIO初始化函数
* @param None
* @retval None
*/
void MX_GPIO_Init(void)
{
GPIO_InitTypeDef GPIO_InitStruct = {0};
/* 使能GPIO时钟 */
__HAL_RCC_GPIOC_CLK_ENABLE();
__HAL_RCC_GPIOD_CLK_ENABLE();
__HAL_RCC_GPIOA_CLK_ENABLE();
__HAL_RCC_GPIOB_CLK_ENABLE();
/* 配置PC13引脚为输出 */
HAL_GPIO_WritePin(GPIOC, GPIO_PIN_13, GPIO_PIN_RESET);
GPIO_InitStruct.Pin = GPIO_PIN_13;
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOC, &GPIO_InitStruct);
/* 配置PA0引脚为模拟输入(气体传感器) */
GPIO_InitStruct.Pin = GPIO_PIN_0;
GPIO_InitStruct.Mode = GPIO_MODE_ANALOG;
GPIO_InitStruct.Pull = GPIO_NOPULL;
HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
}
/**
* @brief I2C1初始化函数
* @param None
* @retval None
*/
void MX_I2C1_Init(void)
{
hi2c1.Instance = I2C1;
hi2c1.Init.ClockSpeed = 100000;
hi2c1.Init.DutyCycle = I2C_DUTYCYCLE_2;
hi2c1.Init.OwnAddress1 = 0;
hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT;
hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE;
hi2c1.Init.OwnAddress2 = 0;
hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE;
hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE;
HAL_I2C_Init(&hi2c1);
}
/**
* @brief ADC1初始化函数
* @param None
* @retval None
*/
void MX_ADC1_Init(void)
{
ADC_ChannelConfTypeDef sConfig = {0};
/* ADC1基本配置 */
hadc1.Instance = ADC1;
hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
hadc1.Init.ContinuousConvMode = DISABLE;
hadc1.Init.DiscontinuousConvMode = DISABLE;
hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
hadc1.Init.NbrOfConversion = 1;
HAL_ADC_Init(&hadc1);
/* 配置ADC通道0(PA0引脚)- 气体传感器 */
sConfig.Channel = ADC_CHANNEL_0;
sConfig.Rank = ADC_REGULAR_RANK_1;
sConfig.SamplingTime = ADC_SAMPLETIME_55CYCLES_5;
HAL_ADC_ConfigChannel(&hadc1, &sConfig);
/* 校准ADC */
HAL_ADCEx_Calibration_Start(&hadc1);
}
/**
* @brief USART1初始化函数
* @param None
* @retval None
*/
void MX_USART1_UART_Init(void)
{
huart1.Instance = USART1;
huart1.Init.BaudRate = 9600;
huart1.Init.WordLength = UART_WORDLENGTH_8B;
huart1.Init.StopBits = UART_STOPBITS_1;
huart1.Init.Parity = UART_PARITY_NONE;
huart1.Init.Mode = UART_MODE_TX_RX;
huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;
huart1.Init.OverSampling = UART_OVERSAMPLING_16;
HAL_UART_Init(&huart1);
/* 启用UART接收中断 */
HAL_NVIC_SetPriority(USART1_IRQn, 0, 0);
HAL_NVIC_EnableIRQ(USART1_IRQn);
}
/**
* @brief 错误处理函数
* @retval None
*/
void Error_Handler(void)
{
__disable_irq();
while (1)
{
}
}
#ifdef USE_FULL_ASSERT
void assert_failed(uint8_t *file, uint32_t line)
{
/* 用户可以添加自己的实现来报告文件名和行号 */
}
#endif /* USE_FULL_ASSERT */
7.原理图
欢迎在评论区交流讨论!