当前位置: 首页 > news >正文

基于STM32开发的智能农业环境监测系统

目录

  1. 引言
  2. 环境准备工作
    • 硬件准备
    • 软件安装与配置
  3. 系统设计
    • 系统架构
    • 硬件连接
  4. 代码实现
    • 初始化代码
    • 控制代码
  5. 应用场景
    • 农田环境监测
    • 温室环境控制
  6. 常见问题及解决方案
    • 常见问题
    • 解决方案
  7. 结论

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开发板中。

步骤:

  1. 下载并安装Keil uVision。
  2. 下载并安装STM32CubeMX。
  3. 下载并安装ST-Link Utility。

3. 系统设计

系统架构

智能农业环境监测系统通过STM32微控制器连接土壤湿度传感器、温度传感器、光照传感器、CO2传感器、OLED显示屏、Wi-Fi模块和数据存储模块,实现对农业环境的实时监测和数据存储,并通过Wi-Fi模块实现远程监控和数据上传。系统包括环境监测模块、数据处理与显示模块、数据存储模块、用户交互模块和远程通信模块。

硬件连接

  1. 将土壤湿度传感器的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,数据引脚连接到STM32的ADC引脚(例如PA0)。
  2. 将温度传感器的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,数据引脚连接到STM32的GPIO引脚(例如PA1)。
  3. 将光照传感器的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,SCL和SDA引脚连接到STM32的I2C引脚(例如PB6、PB7)。
  4. 将CO2传感器的VCC引脚连接到STM32的5V引脚,GND引脚连接到GND,数据引脚连接到STM32的ADC引脚(例如PA2)。
  5. 将OLED显示屏的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,SCL和SDA引脚连接到STM32的I2C引脚(例如PB6、PB7)。
  6. 将数据存储模块的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,CS、MOSI、MISO、SCK引脚分别连接到STM32的SPI引脚(例如PA4、PA5、PA6、PA7)。
  7. 将Wi-Fi模块的TX、RX引脚分别连接到STM32的USART引脚(例如PA9、PA10),VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND。
  8. 将按钮的一个引脚连接到STM32的GPIO引脚(例如PA3),另一个引脚连接到GND。
  9. 将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. 常见问题及解决方案

常见问题

  1. 传感器数据不准确
  2. SD卡无法存储数据
  3. Wi-Fi连接不稳定或数据上传失败

解决方案

  1. 校准传感器
    • 使用已知条件校准各类传感器,确保测量数据的准确性。
  2. 检查SD卡格式
    • 确保SD卡已正确格式化为FAT32,并确认SD卡模块与STM32的连接是否正常。
  3. 优化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…...

浅谈 React Hooks

React Hooks 是 React 16.8 引入的一组 API,用于在函数组件中使用 state 和其他 React 特性(例如生命周期方法、context 等)。Hooks 通过简洁的函数接口,解决了状态与 UI 的高度解耦,通过函数式编程范式实现更灵活 Rea…...

Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以?

Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以? 在 Golang 的面试中,map 类型的使用是一个常见的考点,其中对 key 类型的合法性 是一道常被提及的基础却很容易被忽视的问题。本文将带你深入理解 Golang 中…...

AI Agent与Agentic AI:原理、应用、挑战与未来展望

文章目录 一、引言二、AI Agent与Agentic AI的兴起2.1 技术契机与生态成熟2.2 Agent的定义与特征2.3 Agent的发展历程 三、AI Agent的核心技术栈解密3.1 感知模块代码示例:使用Python和OpenCV进行图像识别 3.2 认知与决策模块代码示例:使用OpenAI GPT-3进…...

阿里云ACP云计算备考笔记 (5)——弹性伸缩

目录 第一章 概述 第二章 弹性伸缩简介 1、弹性伸缩 2、垂直伸缩 3、优势 4、应用场景 ① 无规律的业务量波动 ② 有规律的业务量波动 ③ 无明显业务量波动 ④ 混合型业务 ⑤ 消息通知 ⑥ 生命周期挂钩 ⑦ 自定义方式 ⑧ 滚的升级 5、使用限制 第三章 主要定义 …...

深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法

深入浅出:JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中,随机数的生成看似简单,却隐藏着许多玄机。无论是生成密码、加密密钥,还是创建安全令牌,随机数的质量直接关系到系统的安全性。Jav…...

STM32标准库-DMA直接存储器存取

文章目录 一、DMA1.1简介1.2存储器映像1.3DMA框图1.4DMA基本结构1.5DMA请求1.6数据宽度与对齐1.7数据转运DMA1.8ADC扫描模式DMA 二、数据转运DMA2.1接线图2.2代码2.3相关API 一、DMA 1.1简介 DMA(Direct Memory Access)直接存储器存取 DMA可以提供外设…...

tree 树组件大数据卡顿问题优化

问题背景 项目中有用到树组件用来做文件目录,但是由于这个树组件的节点越来越多,导致页面在滚动这个树组件的时候浏览器就很容易卡死。这种问题基本上都是因为dom节点太多,导致的浏览器卡顿,这里很明显就需要用到虚拟列表的技术&…...

Java 二维码

Java 二维码 **技术&#xff1a;**谷歌 ZXing 实现 首先添加依赖 <!-- 二维码依赖 --><dependency><groupId>com.google.zxing</groupId><artifactId>core</artifactId><version>3.5.1</version></dependency><de…...

论文笔记——相干体技术在裂缝预测中的应用研究

目录 相关地震知识补充地震数据的认识地震几何属性 相干体算法定义基本原理第一代相干体技术&#xff1a;基于互相关的相干体技术&#xff08;Correlation&#xff09;第二代相干体技术&#xff1a;基于相似的相干体技术&#xff08;Semblance&#xff09;基于多道相似的相干体…...

Linux离线(zip方式)安装docker

目录 基础信息操作系统信息docker信息 安装实例安装步骤示例 遇到的问题问题1&#xff1a;修改默认工作路径启动失败问题2 找不到对应组 基础信息 操作系统信息 OS版本&#xff1a;CentOS 7 64位 内核版本&#xff1a;3.10.0 相关命令&#xff1a; uname -rcat /etc/os-rele…...