基于STM32开发的智能农业环境监测系统
目录
- 引言
- 环境准备工作
- 硬件准备
- 软件安装与配置
- 系统设计
- 系统架构
- 硬件连接
- 代码实现
- 初始化代码
- 控制代码
- 应用场景
- 农田环境监测
- 温室环境控制
- 常见问题及解决方案
- 常见问题
- 解决方案
- 结论
1. 引言
智能农业环境监测系统通过集成多种环境传感器,实时监测土壤湿度、温度、光照、CO2浓度等环境参数,帮助农民和农业工作者优化农田或温室的管理,提升农作物的产量和质量。本文将介绍如何使用STM32微控制器设计和实现一个智能农业环境监测系统。
2. 环境准备工作
硬件准备
- STM32开发板(例如STM32F103C8T6)
- 土壤湿度传感器(例如YL-69)
- 温度传感器(例如DS18B20)
- 光照传感器(例如BH1750)
- CO2传感器(例如MG-811)
- OLED显示屏(用于显示环境数据)
- Wi-Fi模块(例如ESP8266,用于远程监控)
- 数据存储模块(例如SD卡模块)
- 按钮和LED(用于用户交互)
- 面包板和连接线
- USB下载线
软件安装与配置
- Keil uVision:用于编写、编译和调试代码。
- STM32CubeMX:用于配置STM32微控制器的引脚和外设。
- ST-Link Utility:用于将编译好的代码下载到STM32开发板中。
步骤:
- 下载并安装Keil uVision。
- 下载并安装STM32CubeMX。
- 下载并安装ST-Link Utility。
3. 系统设计
系统架构
智能农业环境监测系统通过STM32微控制器连接土壤湿度传感器、温度传感器、光照传感器、CO2传感器、OLED显示屏、Wi-Fi模块和数据存储模块,实现对农业环境的实时监测和数据存储,并通过Wi-Fi模块实现远程监控和数据上传。系统包括环境监测模块、数据处理与显示模块、数据存储模块、用户交互模块和远程通信模块。
硬件连接
- 将土壤湿度传感器的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,数据引脚连接到STM32的ADC引脚(例如PA0)。
- 将温度传感器的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,数据引脚连接到STM32的GPIO引脚(例如PA1)。
- 将光照传感器的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,SCL和SDA引脚连接到STM32的I2C引脚(例如PB6、PB7)。
- 将CO2传感器的VCC引脚连接到STM32的5V引脚,GND引脚连接到GND,数据引脚连接到STM32的ADC引脚(例如PA2)。
- 将OLED显示屏的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,SCL和SDA引脚连接到STM32的I2C引脚(例如PB6、PB7)。
- 将数据存储模块的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,CS、MOSI、MISO、SCK引脚分别连接到STM32的SPI引脚(例如PA4、PA5、PA6、PA7)。
- 将Wi-Fi模块的TX、RX引脚分别连接到STM32的USART引脚(例如PA9、PA10),VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND。
- 将按钮的一个引脚连接到STM32的GPIO引脚(例如PA3),另一个引脚连接到GND。
- 将LED的正极引脚连接到STM32的GPIO引脚(例如PA4),负极引脚连接到GND。
4. 代码实现
初始化代码
#include "stm32f1xx_hal.h"
#include "soil_moisture.h"
#include "temperature.h"
#include "light_sensor.h"
#include "co2_sensor.h"
#include "oled.h"
#include "sd_card.h"
#include "wifi.h"
#include "button.h"
#include "led.h"void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_USART1_UART_Init(void);
static void MX_I2C1_Init(void);
static void MX_SPI1_Init(void);
static void MX_ADC1_Init(void);int main(void) {HAL_Init();SystemClock_Config();MX_GPIO_Init();MX_USART1_UART_Init();MX_I2C1_Init();MX_SPI1_Init();MX_ADC1_Init();SoilMoisture_Init();Temperature_Init();LightSensor_Init();CO2Sensor_Init();OLED_Init();SDCard_Init();WiFi_Init();Button_Init();LED_Init();while (1) {float soilMoisture = SoilMoisture_Read();float temperature = Temperature_Read();uint16_t lightLevel = LightSensor_Read();float co2Level = CO2Sensor_Read();char displayStr[64];sprintf(displayStr, "Soil: %.2f%%\nTemp: %.2fC\nLight: %d lx\nCO2: %.2f ppm", soilMoisture, temperature, lightLevel, co2Level);OLED_DisplayString(displayStr);SDCard_SaveData(soilMoisture, temperature, lightLevel, co2Level);if (WiFi_IsConnected()) {WiFi_SendData(soilMoisture, temperature, lightLevel, co2Level);}if (Button_IsPressed()) {LED_On();HAL_Delay(1000);LED_Off();}HAL_Delay(5000);}
}void SystemClock_Config(void) {// 配置系统时钟
}static void MX_GPIO_Init(void) {// 初始化GPIO__HAL_RCC_GPIOA_CLK_ENABLE();GPIO_InitTypeDef GPIO_InitStruct = {0};GPIO_InitStruct.Pin = GPIO_PIN_0 | GPIO_PIN_1 | GPIO_PIN_2 | GPIO_PIN_3 | GPIO_PIN_4 | GPIO_PIN_5;GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;GPIO_InitStruct.Pull = GPIO_NOPULL;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
}static void MX_USART1_UART_Init(void) {// 初始化USART1用于Wi-Fi通信huart1.Instance = USART1;huart1.Init.BaudRate = 115200;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;if (HAL_UART_Init(&huart1) != HAL_OK) {Error_Handler();}
}static void MX_I2C1_Init(void) {// 初始化I2C1用于OLED显示屏通信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;if (HAL_I2C_Init(&hi2c1) != HAL_OK) {Error_Handler();}
}static void MX_SPI1_Init(void) {// 初始化SPI1用于SD卡通信hspi1.Instance = SPI1;hspi1.Init.Mode = SPI_MODE_MASTER;hspi1.Init.Direction = SPI_DIRECTION_2LINES;hspi1.Init.DataSize = SPI_DATASIZE_8BIT;hspi1.Init.CLKPolarity = SPI_POLARITY_LOW;hspi1.Init.CLKPhase = SPI_PHASE_1EDGE;hspi1.Init.NSS = SPI_NSS_SOFT;hspi1.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_16;hspi1.Init.FirstBit = SPI_FIRSTBIT_MSB;hspi1.Init.TIMode = SPI_TIMODE_DISABLE;hspi1.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE;hspi1.Init.CRCPolynomial = 10;if (HAL_SPI_Init(&hspi1) != HAL_OK) {Error_Handler();}
}static void MX_ADC1_Init(void) {// 初始化ADC1用于传感器数据采集ADC_ChannelConfTypeDef sConfig = {0};hadc1.Instance = ADC1;hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;hadc1.Init.ContinuousConvMode = ENABLE;hadc1.Init.DiscontinuousConvMode = DISABLE;hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;hadc1.Init.NbrOfConversion = 1;if (HAL_ADC_Init(&hadc1) != HAL_OK) {Error_Handler();}sConfig.Channel = ADC_CHANNEL_0;sConfig.Rank = ADC_REGULAR_RANK_1;sConfig.SamplingTime = ADC_SAMPLETIME_55CYCLES_5;if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK) {Error_Handler();}HAL_ADC_Start(&hadc1);
}
控制代码
#include "soil_moisture.h"
#include "temperature.h"
#include "light_sensor.h"
#include "co2_sensor.h"
#include "oled.h"
#include "sd_card.h"
#include "wifi.h"
#include "button.h"
#include "led.h"void SoilMoisture_Init(void) {// 初始化土壤湿度传感器
}float SoilMoisture_Read(void) {// 读取土壤湿度数据
}void Temperature_Init(void) {// 初始化温度传感器
}float Temperature_Read(void) {// 读取温度数据
}void LightSensor_Init(void) {// 初始化光照传感器
}uint16_t LightSensor_Read(void) {// 读取光照数据
}void CO2Sensor_Init(void) {// 初始化CO2传感器
}float CO2Sensor_Read(void) {// 读取CO2浓度数据
}void OLED_Init(void) {// 初始化OLED显示屏
}void OLED_DisplayString(char *str) {// 在OLED显示屏上显示字符串
}void SDCard_Init(void) {// 初始化SD卡模块
}void SDCard_SaveData(float soilMoisture, float temperature, uint16_t lightLevel, float co2Level) {// 将数据保存到SD卡
}void WiFi_Init(void) {// 初始化Wi-Fi模块
}bool WiFi_IsConnected(void) {// 检查Wi-Fi是否已连接
}void WiFi_SendData(float soilMoisture, float temperature, uint16_t lightLevel, float co2Level) {// 发送数据到服务器
}void Button_Init(void) {// 初始化按钮
}bool Button_IsPressed(void) {// 检测按钮是否按下
}void LED_Init(void) {// 初始化LED
}void LED_On(void) {// 打开LED
}void LED_Off(void) {// 关闭LED
}
⬇帮大家整理了单片机的资料
包括stm32的项目合集【源码+开发文档】
点击下方蓝字即可领取,感谢支持!⬇
点击领取更多嵌入式详细资料
问题讨论,stm32的资料领取可以私信!
5. 应用场景
农田环境监测
本系统可以应用于农田环境的实时监测,通过采集和分析环境数据,帮助农民及时调整灌溉、施肥和通风策略,优化农作物的生长环境。
温室环境控制
本系统还可以应用于温室环境的智能控制,实时监测温室内的温度、湿度、光照和CO2浓度,通过联网实现远程监控和自动化管理,提升温室种植的效率和产量。
6. 常见问题及解决方案
常见问题
- 传感器数据不准确
- SD卡无法存储数据
- Wi-Fi连接不稳定或数据上传失败
解决方案
- 校准传感器
- 使用已知条件校准各类传感器,确保测量数据的准确性。
- 检查SD卡格式
- 确保SD卡已正确格式化为FAT32,并确认SD卡模块与STM32的连接是否正常。
- 优化Wi-Fi设置
- 检查Wi-Fi信号强度和网络配置,确保Wi-Fi模块与路由器的连接稳定。
7. 结论
本文介绍了如何使用STM32微控制器和多种传感器实现一个智能农业环境监测系统,从硬件准备、环境配置到代码实现,详细介绍了每一步的操作步骤。通过本文的学习,读者可以掌握基本的嵌入式开发技能,并将其应用到农业环境监测和温室控制项目中。
相关文章:

基于STM32开发的智能农业环境监测系统
目录 引言环境准备工作 硬件准备软件安装与配置系统设计 系统架构硬件连接代码实现 初始化代码控制代码应用场景 农田环境监测温室环境控制常见问题及解决方案 常见问题解决方案结论 1. 引言 智能农业环境监测系统通过集成多种环境传感器,实时监测土壤湿度、温度…...

【SQL】平均售价
目录 题目 分析 代码 题目 表:Prices ------------------------ | Column Name | Type | ------------------------ | product_id | int | | start_date | date | | end_date | date | | price | int | ---------------…...

存储器与CPU的连接
1.单块存储芯片与CPU的连接 单独的一块独立的存储芯片提供的线有:地址总线,数据总线,读写控制线,片选线,如果该存储器只有八根数据总线用于输出数据,而cpu一次可以读64位的数据呢? 我们可以将八…...
unity--webgl 访问本地index.html
目录 1:使用本地服务器 1.1 使用 Python 的 SimpleHTTPServer 1.2 使用 Node.js 的 http-server 2:让其他人通过 IP 地址来访问你的 Unity WebGL 项目 2.1: 确保服务器可访问 2.2 获取公共 IP 地址 2.3 配置本地服务器 1.使用 Python 的 SimpleHTTPServer 2…...
慢慢欣赏DPDK RTE_MAX_ETHPORTS的定义
DPDK代码里面,RTE_MAX_ETHPORTS是一个常见的宏定义,但是在.c和.h文件找不到其定义,在全文件搜索条件下,在config/meson.build找到这么一个定义 dpdk_conf.set(RTE_MAX_ETHPORTS, get_option(max_ethports)) 该宏定义是根据构建输…...

Java Nacos与Gateway的使用
Java系列文章目录 IDEA使用指南 Java泛型总结(快速上手详解) Java Lambda表达式总结(快速上手详解) Java Optional容器总结(快速上手图解) Java 自定义注解笔记总结(油管) Jav…...

前端项目中的Server-sent Events(SSE)项目实践及其与websocket的区别
前端项目中的Server-sent Events(SSE)项目实践 前言 在前端开发中,实时数据更新是提升用户体验的重要因素之一。Server-SentEvents(SSE)是一种高效的技术,允许服务器通过单向连接将实时数据推送到客户端。下面将从SSE的基本改变,使用场景展…...

《老俞闲话|唯爱和热情不可辜负》读后感
《老俞闲话|唯爱和热情不可辜负》读后感 俞敏洪先生的这篇讲话充满了深情与智慧,他以自己丰富的人生经历和教育实践,向我们展现了一位教育家对于教育事业的热爱和对教师角色的深刻理解。 情感真挚,触动人心 俞敏洪先生的讲话中流…...

C语言 ——— 在杨氏矩阵中查找具体的某个数
目录 何为杨氏矩阵 题目要求 代码实现 何为杨氏矩阵 可以把杨氏矩阵理解为一个二维数组,这个二维数组中的每一行从左到右是递增的,每一列从上到下是递增的 题目要求 在杨氏矩阵中查找具体的某个数 要求:时间复杂度小于O(N) 代码实现…...

DAI-Net: 基于对偶自适应交互网络的药物推荐算法
引言 DAI-Net: Dual Adaptive Interaction Network for Coordinated Medication Recommendation 论文链接:https://ieeexplore.ieee.org/document/10614809 代码链接:GitHub - obananas/DAI-Net 在现代医疗保健中,如何利用电子健康记录&a…...

haproxy高级功能及配置
章节 一、haproxy 基础用法 二、haproxy 高级用法 三、haproxy之ACL的使用 目录 1 基于cookie的会话保持 1.1 cookie命名,并赋予其值 1.2 验证cookie信息 1.2.1 Windows浏览器验证 1.2.2 Linux下虚拟机验证 2 IP透传 2.1 四层与七层透传的区别 2.2 七层IP透传 2.2…...
【前端】NodeJS:记账本案例优化(MongoDB数据库)
文章目录 1 字符串转为时间对象——Moment2 记账本实例优化 1 字符串转为时间对象——Moment Moment.js中文网:https://momentjs.cn/docs/#/parsing/。 npm install moment // 安装moment var moment require(moment); // require moment().format(); 2 记账本实…...
Padding Mask;Sequence Mask;为什么如果没有适当的掩码机制,解码器在生成某个位置的输出时,可能会“看到”并错误地利用该位置之后的信息
目录 掩码Mask Padding Mask Sequence Mask 为什么需要Sequence Mask? Sequence Mask是如何工作的? 具体实现 为什么如果没有适当的掩码机制,解码器在生成某个位置的输出时,可能会“看到”并错误地利用该位置之后的信息 自回归性质 一、定义 二、性质 三、应用限制…...
派森学长带你学python—字典
一.字典的创建与删除 字典类型是根据一个信息查找另一个信息的方式构成了键值对 字典和列表均为可变数据类型,可变数据类型具有增删改等操作 字典中的键唯一,值可以有多个相同的;字典中的键要求是不可变序列,如字符串、整数、浮…...

如何设置 Visual Studio Code 的滚轮缩放功能
Visual Studio Code (VSCode) 是一个强大的代码编辑器,提供了许多便捷的功能来提高开发效率。其中之一就是通过滚轮缩放字体大小。以下是详细的设置步骤: 步骤 1:打开设置页面 首先,启动 Visual Studio Code。在左上角点击 “文…...

Python模拟退火算法
目录 模拟退火算法简介模拟退火算法的步骤模拟退火算法的Python实现场景:函数优化问题 代码解释总结 模拟退火算法简介 模拟退火算法(Simulated Annealing, SA)是一种基于物理退火过程的随机搜索算法,用于寻找全局最优解。其灵感…...

C语言典型例题36
《C程序设计教程(第四版)——谭浩强》 例题3.4 输入一个字符,判别它是否为大写字母,如果是,将它转换为小写字母:如果不是,不转换。然后输出最后要输出的字符。 代码: //《C程序设计…...

实现高亮的全文分页检索
文章目录 🌞 Sun Frame:SpringBoot 的轻量级开发框架(个人开源项目推荐)🌟 亮点功能📦 spring cloud模块概览常用工具 🔗 更多信息1.sun-club-infra 模块SubjectEsServiceImpl.java1.querySubje…...
【buildroot与yocto区别】
buildroot与yocto区别 Buildroot和Yocto的主要区别在于它们的使用目的、构建过程、以及输出的内容。 使用目的:Buildroot主要用于构建根文件系统,而Yocto项目则用于帮助开发人员为嵌入式产品创建定制的基于Linux的系统。Yocto项目不仅仅构建根文件系…...

原创音乐小程序的设计
管理员账户功能包括:系统首页,个人中心,用户管理,歌曲类型管理,歌曲信息管理,热门歌手管理,音乐资讯管理,系统管理 微信端账号功能包括:系统首页,歌曲信息&a…...
Spring Boot 实现流式响应(兼容 2.7.x)
在实际开发中,我们可能会遇到一些流式数据处理的场景,比如接收来自上游接口的 Server-Sent Events(SSE) 或 流式 JSON 内容,并将其原样中转给前端页面或客户端。这种情况下,传统的 RestTemplate 缓存机制会…...

循环冗余码校验CRC码 算法步骤+详细实例计算
通信过程:(白话解释) 我们将原始待发送的消息称为 M M M,依据发送接收消息双方约定的生成多项式 G ( x ) G(x) G(x)(意思就是 G ( x ) G(x) G(x) 是已知的)࿰…...

srs linux
下载编译运行 git clone https:///ossrs/srs.git ./configure --h265on make 编译完成后即可启动SRS # 启动 ./objs/srs -c conf/srs.conf # 查看日志 tail -n 30 -f ./objs/srs.log 开放端口 默认RTMP接收推流端口是1935,SRS管理页面端口是8080,可…...

智能仓储的未来:自动化、AI与数据分析如何重塑物流中心
当仓库学会“思考”,物流的终极形态正在诞生 想象这样的场景: 凌晨3点,某物流中心灯火通明却空无一人。AGV机器人集群根据实时订单动态规划路径;AI视觉系统在0.1秒内扫描包裹信息;数字孪生平台正模拟次日峰值流量压力…...
2023赣州旅游投资集团
单选题 1.“不登高山,不知天之高也;不临深溪,不知地之厚也。”这句话说明_____。 A、人的意识具有创造性 B、人的认识是独立于实践之外的 C、实践在认识过程中具有决定作用 D、人的一切知识都是从直接经验中获得的 参考答案: C 本题解…...

RSS 2025|从说明书学习复杂机器人操作任务:NUS邵林团队提出全新机器人装配技能学习框架Manual2Skill
视觉语言模型(Vision-Language Models, VLMs),为真实环境中的机器人操作任务提供了极具潜力的解决方案。 尽管 VLMs 取得了显著进展,机器人仍难以胜任复杂的长时程任务(如家具装配),主要受限于人…...

实战三:开发网页端界面完成黑白视频转为彩色视频
一、需求描述 设计一个简单的视频上色应用,用户可以通过网页界面上传黑白视频,系统会自动将其转换为彩色视频。整个过程对用户来说非常简单直观,不需要了解技术细节。 效果图 二、实现思路 总体思路: 用户通过Gradio界面上…...
如何配置一个sql server使得其它用户可以通过excel odbc获取数据
要让其他用户通过 Excel 使用 ODBC 连接到 SQL Server 获取数据,你需要完成以下配置步骤: ✅ 一、在 SQL Server 端配置(服务器设置) 1. 启用 TCP/IP 协议 打开 “SQL Server 配置管理器”。导航到:SQL Server 网络配…...

快速排序算法改进:随机快排-荷兰国旗划分详解
随机快速排序-荷兰国旗划分算法详解 一、基础知识回顾1.1 快速排序简介1.2 荷兰国旗问题 二、随机快排 - 荷兰国旗划分原理2.1 随机化枢轴选择2.2 荷兰国旗划分过程2.3 结合随机快排与荷兰国旗划分 三、代码实现3.1 Python实现3.2 Java实现3.3 C实现 四、性能分析4.1 时间复杂度…...
算法250609 高精度
加法 #include<stdio.h> #include<iostream> #include<string.h> #include<math.h> #include<algorithm> using namespace std; char input1[205]; char input2[205]; int main(){while(scanf("%s%s",input1,input2)!EOF){int a[205]…...