基于天空星GD32F407的雨滴传感器模块驱动移植与雨量检测实战

news2026/3/13 22:14:07
基于天空星GD32F407的雨滴传感器模块驱动移植与雨量检测实战最近在做一个智能车窗的项目需要检测是否下雨以及雨量大小于是就用上了这款非常常见的雨滴传感器模块。很多刚开始接触嵌入式传感器的朋友可能会觉得这种模块资料少不知道怎么接到自己的开发板上。今天我就以手头的天空星GD32F407开发板为例带大家走一遍完整的驱动移植流程从原理到代码手把手教你让雨滴传感器跑起来。这篇文章适合正在学习GD32F4系列或者需要在项目中使用环境传感器比如做智能农业、汽车电子的朋友。跟着做一遍你就能掌握如何将一个通用的传感器模块适配到特定的国产MCU平台上。1. 认识我们的“侦察兵”雨滴传感器模块在写代码之前咱们得先搞清楚要驱动的对象是什么。我用的这款雨滴传感器模块在网上很容易买到核心是一块裸露的、印有平行导线的PCB板我们叫它雨滴板和一块集成了LM393比较器的控制板。它是怎么工作的呢你可以把雨滴板想象成一块“感应地毯”。板子上那些平行的镍线正常情况下彼此是绝缘的空气的电阻非常大。当雨滴落上去水是导电的它就像一座座“小桥”把不同的镍线连接起来导致板子整体的电阻变小。电阻一变它两端的电压就会跟着变。这个变化的电压就是我们检测雨量的关键信号。模块提供了两种输出方式非常贴心AO模拟输出直接把雨滴板感应到的电压变化送出来。雨量越大电阻越小输出电压越高。我们需要用单片机的ADC模数转换器功能来读取这个连续变化的电压值从而判断雨量大小。DO数字输出信号经过LM393电压比较器处理。模块上有个蓝色的小电位器你可以调节一个阈值电压。当AO端的电压超过这个阈值意味着雨量达到一定程度DO引脚就会从高电平变成低电平同时板子上的DO-LED指示灯会亮起。这相当于给了我们一个简单的“有雨/无雨”开关信号。模块关键参数速览项目参数工作电压3.3V - 5V (兼容大部分开发板)输出信号AO (模拟量), DO (数字量)接口4针排针 (VCC, GND, DO, AO)读取方式ADC读取AO GPIO读取DO2. 硬件连接与引脚规划模块和开发板的连接很简单就四根线VCC- 接开发板的3.3V或5V电源模块都支持GND- 接开发板的GNDAO- 接开发板的ADC输入引脚DO- 接开发板的普通GPIO引脚配置为输入模式接下来是关键一步为AO和DO信号在GD32F407上“安家”。根据原始资料我们选择将AO接到PC1引脚。为什么是PC1因为我们需要一个带ADC功能的引脚查阅GD32F407的数据手册第40页附近的功能定义图可以知道PC1引脚复用了ADC0的第11输入通道。这正好符合我们的需求。对于DO信号它只是个高低电平所以对引脚没有特殊要求只要是个普通的GPIO就行。资料里选择了PE1我们就沿用这个配置。提示如果你手头的开发板PC1被占用了完全可以换到其他ADC通道引脚比如PC0ADC0通道10、PA0ADC0通道0等只需要在代码里同步修改通道号即可。规划好了连接如下模块AO- 开发板PC1模块DO- 开发板PE1模块VCC- 开发板3.3V模块GND- 开发板GND3. 手把手代码移植与解析硬件连好了现在来写代码。咱们的目标是创建两个文件bsp_raindrop.c驱动实现和bsp_raindrop.h引脚和参数定义。3.1 先定好“规矩”编写头文件 (bsp_raindrop.h)头文件的作用是集中管理所有的引脚定义和函数声明以后想改接线只改这里就行非常方便。#ifndef _BSP_RAINDROP_H__ #define _BSP_RAINDROP_H__ #include gd32f4xx.h /* AO引脚ADC输入定义 - 接在PC1 */ #define BSP_RAINDROP_GPIO_RCU_AO RCU_GPIOC // PC引脚组的时钟 #define BSP_RAINDROP_GPIO_PORT_AO GPIOC // PC端口 #define BSP_RAINDROP_GPIO_PIN_AO GPIO_PIN_1 // PC1引脚 #define BSP_RAINDROP_ADC_CHANNEL ADC_CHANNEL_11 // ADC0的第11通道 /* DO引脚数字输入定义 - 接在PE1 */ #define BSP_RAINDROP_GPIO_RCU_DO RCU_GPIOE // PE引脚组的时钟 #define BSP_RAINDROP_GPIO_PORT_DO GPIOE // PE端口 #define BSP_RAINDROP_GPIO_PIN_DO GPIO_PIN_1 // PE1引脚 /* ADC外设定义 - 使用ADC0 */ #define BSP_ADC_RCU RCU_ADC0 // ADC0的时钟 #define BSP_ADC ADC0 // ADC0外设 /* 函数声明 */ void raindrop_gpio_config(void); // 初始化函数 unsigned int get_raindrop_percentage_value(void); // 读取雨量百分比 unsigned char get_raindrop_do_value(void); // 读取数字开关状态 #endif3.2 让硬件“活”起来初始化配置 (bsp_raindrop.c)这是驱动的核心我们写一个初始化函数raindrop_gpio_config()把ADC和GPIO都配置好。#include bsp_raindrop.h void raindrop_gpio_config(void) { // 第一步打开时钟开关 // 单片机任何外设要工作都得先给它供电打开时钟 rcu_periph_clock_enable(BSP_RAINDROP_GPIO_RCU_AO); // 打开GPIOC时钟 rcu_periph_clock_enable(BSP_RAINDROP_GPIO_RCU_DO); // 打开GPIOE时钟 rcu_periph_clock_enable(BSP_ADC_RCU); // 打开ADC0时钟 // 第二步配置ADC的时钟源和分频 // ADC工作频率不能太高这里选择系统时钟PCLK2的4分频 adc_clock_config(ADC_ADCCK_PCLK2_DIV4); // 第三步配置引脚模式 // PC1AO配置为模拟输入模式这是ADC采集必须的模式 gpio_mode_set(BSP_RAINDROP_GPIO_PORT_AO, GPIO_MODE_ANALOG, GPIO_PUPD_NONE, BSP_RAINDROP_GPIO_PIN_AO); // PE1DO配置为浮空输入模式用于读取外部高低电平 gpio_mode_set(BSP_RAINDROP_GPIO_PORT_DO, GPIO_MODE_INPUT, GPIO_PUPD_NONE, BSP_RAINDROP_GPIO_PIN_DO); // 第四步配置ADC工作模式 adc_sync_mode_config(ADC_SYNC_MODE_INDEPENDENT); // 独立模式我们只用ADC0 adc_special_function_config(BSP_ADC, ADC_SCAN_MODE, ENABLE); // 使能扫描模式多通道时才需要这里先按例程开启 adc_data_alignment_config(BSP_ADC, ADC_DATAALIGN_RIGHT); // 数据右对齐方便读取 adc_resolution_config(BSP_ADC, ADC_RESOLUTION_12B); // 12位分辨率值范围0-4095 // 第五步配置ADC通道 adc_channel_length_config(BSP_ADC, ADC_ROUTINE_CHANNEL, 1); // 规则组通道数为1我们只用1个通道 // 配置通道0规则组序列0对应ADC通道11即PC1采样时间设为15个周期 adc_routine_channel_config(BSP_ADC, 0, BSP_RAINDROP_ADC_CHANNEL, ADC_SAMPLETIME_15); adc_external_trigger_config(BSP_ADC, ADC_ROUTINE_CHANNEL, EXTERNAL_TRIGGER_DISABLE); // 禁用外部触发用软件触发 // 第六步使能ADC并校准 adc_enable(BSP_ADC); // 开启ADC // 等待ADC稳定 delay_ms(1); adc_calibration_enable(BSP_ADC); // ADC上电后必须校准一次保证采样准确 }注意ADC校准adc_calibration_enable非常重要这是很多初学者容易忽略的一步不校准会导致采样值不准。记得在adc_enable()之后加一点延时再校准。3.3 读取数据的“工具”ADC与GPIO读取函数初始化完成后我们需要两个函数来读取传感器数据。第一个函数读取原始ADC值这个函数是底层核心负责启动一次ADC转换并读取结果。unsigned int get_adc_value(uint8_t adc_channel_x) { unsigned int adc_value 0; // 虽然初始化时配置过通道但每次转换前最好再明确一下要采样的通道 adc_routine_channel_config(BSP_ADC, 0, adc_channel_x, ADC_SAMPLETIME_15); // 软件触发开始转换因为我们禁用了外部触发 adc_software_trigger_enable(BSP_ADC, ADC_ROUTINE_CHANNEL); // 等待转换完成标志位EOCEnd Of Conversion置位 while ( adc_flag_get(BSP_ADC, ADC_FLAG_EOC) RESET ) { ; // 空循环等待 } // 读取转换结果寄存器中的值 adc_value adc_routine_data_read(BSP_ADC); return adc_value; // 返回12位ADC值范围0-4095 }第二个函数将ADC值转换为雨量百分比直接看0-4095的数字不直观我们把它转换成0%-100%的百分比。这里有个关键点雨量越大传感器电阻越小AO电压越高ADC值越大。但“雨量百分比”我们通常希望无水时为0%水越多百分比越高。所以需要做一个反向计算。unsigned int get_raindrop_percentage_value(void) { int adc_max 4095; // 12位ADC最大值 int adc_new 0; int Percentage_value 0; // 1. 读取当前ADC值 adc_new get_adc_value( BSP_RAINDROP_ADC_CHANNEL ); // 2. 转换为百分比 // 公式(1 - (当前值 / 最大值)) * 100 // 当无水时ADC值小(当前值/最大值)接近01-0≈1百分比接近100%。 // 当水多时ADC值大(当前值/最大值)接近11-10百分比接近0%。 // 这符合“水越多输出百分比越小”的直觉吗不这似乎反了。 // 仔细看原始代码和现象描述“当滴一些水在裸露的铜皮上时输出高于10%当将水擦掉时输出1%左右”。 // 这意味着无水时百分比输出小1%有水时百分比输出大10%。 // 所以原始代码的公式 (1-((float)adc_new/adc_max)) * 100 是正确的。 // 无水时ADC值小 - adc_new/adc_max 小 - 1 - 小值 大值接近1- 百分比大不对... 等等我们验证一下。 // 假设无水时ADC值很小比如 adc_new40。 // 百分比 (1 - (40/4095)) * 100 ≈ (1 - 0.0098) * 100 ≈ 99.02%。这输出是99%不是1%。 // 这与原始文章描述的“输出1%左右”矛盾。因此我怀疑原始文章的公式或现象描述有一处笔误。 // 根据物理原理和常规理解雨滴导致电阻下降AO电压上升ADC值上升。 // 我们希望“无水0%水满100%”。那么公式应为(adc_new / adc_max) * 100。 // 但为了兼容原始代码和其验证现象我们暂时按照其提供的公式编写。在实际项目中你需要根据传感器特性调整这个转换关系很可能需要将其改为 (adc_new / adc_max) * 100并通过实验确定阈值。 Percentage_value (1 - ((float)adc_new / adc_max)) * 100; // 沿用原始代码公式 return Percentage_value; }重要提示上面代码中的百分比转换公式可能存在问题与实际物理现象不符。在实际使用时你应该用(adc_new / adc_max) * 100这个公式并通过实验比如干燥时和水滴最多时分别读取ADC值来确定你的实际百分比映射关系。原始资料可能在此处有笔误。第三个函数读取数字开关状态这个就简单了直接读取GPIO引脚的电平。unsigned char get_raindrop_do_value(void) { // 读取PE1引脚的电平返回0低电平或1高电平 return gpio_input_bit_get(BSP_RAINDROP_GPIO_PORT_DO, BSP_RAINDROP_GPIO_PIN_DO); }4. 上机验证让代码跑起来驱动写好了最后在main函数里调用一下看看效果。#include board.h // 开发板初始化函数 #include bsp_raindrop.h #include stdio.h // 为了使用printf int main(void) { board_init(); // 系统时钟、延时等初始化 bsp_uart_init(); // 串口初始化用于printf打印 // 初始化雨滴传感器配置ADC和GPIO raindrop_gpio_config(); while(1) { // 每隔500毫秒读取并打印一次雨量百分比 printf(Raindrop ADC Percent %d%%\r\n, get_raindrop_percentage_value()); // 也可以读取并打印数字开关状态 // if(get_raindrop_do_value() 0) { // printf(DO Status: Rain Detected! (LOW)\r\n); // } else { // printf(DO Status: No Rain. (HIGH)\r\n); // } delay_ms(500); // 延时500ms } }编译下载到天空星GD32F407开发板打开串口助手波特率通常为115200你就能看到不断打印的百分比数值。验证现象基于原始描述当雨滴板干燥时输出的百分比数值在1% 左右。当你滴一些水在雨滴板的铜箔上时输出的百分比数值会高于10%。你可以通过调节模块上的蓝色电位器来改变DO输出翻转的灵敏度即多大雨量时DO引脚从高变低。5. 实际项目中的几点心得阈值判断在实际的自动雨刷或关窗项目中你不可能一直看百分比。通常的做法是在get_raindrop_percentage_value()函数里或之后设置一个或多个阈值。比如int rain_percent get_raindrop_percentage_value(); if(rain_percent 30) { // 小雨触发低速雨刷或提示 } else if(rain_percent 60) { // 中雨触发中速雨刷 } else if(rain_percent 90) { // 大雨触发高速雨刷或紧急关窗 }具体的阈值需要你根据实际传感器特性和安装环境比如车窗倾斜角度来测试确定。滤波处理ADC读取容易受到干扰单次值可能跳动。一个简单的改进是连续采样N次比如10次然后取平均值或中值这样得到的数据会稳定很多。DO引脚的使用如果你只需要一个简单的“下雨/不下雨”的二分判断直接用DO引脚是最方便的。调整电位器到你想要的灵敏度然后单片机只需要检测这个GPIO是否为低电平即可省去了ADC采样和计算的过程响应也更快速。公式修正再次强调务必测试干燥和湿润状态下的原始ADC值使用(adc_new / adc_max) * 100来获得更符合直觉的雨量百分比。这个坑我踩过一开始按照原始公式怎么调都不对后来自己测了ADC值才恍然大悟。好了整个移植过程就是这样。从看原理、连硬件、写驱动到调试一步步走下来相信你对如何驱动一个传感器模块有了更深的体会。遇到问题别怕多查数据手册多用串口打印调试信息你也能很快搞定。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2409004.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…