呼延冰枫 发表于 2025-8-25 15:23:45

基于stm32单片机家庭环境监测系统

 
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本地实时显示所有传感器数据及系统状态​温湿度传感器​AHT20I2C高精度测量环境温度和湿度​光照传感器​GY-30 (BH1750)I2C检测环境光照强度 (Lux)​气体传感器​MQ-135模拟/数字综合检测空气质量、CO2、CO、烟雾、苯等有害气体​人体红外传感器​HC-SR501数字GPIO检测人体移动​蓝牙模块​HC-05 或 HC-06UART与手机小程序进行串口蓝牙通信​Wi-Fi模块​ESP-01S (ESP8266) 或 ESP32UART预留物联网扩展接口,可接入云平台​声学提示​有源蜂鸣器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);

/**
* @briefIR传感器初始化 - 简化版
* @paramNone
* @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检测人体存在 - 简化版
* @paramNone
* @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;
      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;
      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;
      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;
      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)
{
}
}

#ifdefUSE_FULL_ASSERT
void assert_failed(uint8_t *file, uint32_t line)
{
/* 用户可以添加自己的实现来报告文件名和行号 */
}
#endif /* USE_FULL_ASSERT */ 
7.​原理图



 

欢迎在评论区交流讨论!​
 
来源:豆瓜网用户自行投稿发布,如果侵权,请联系站长删除
页: [1]
查看完整版本: 基于stm32单片机家庭环境监测系统