GD32E230C8T6驱动EC11旋转编码器:硬件连接、消抖算法与方向判断实战
GD32E230C8T6驱动EC11旋转编码器硬件连接、消抖算法与方向判断实战最近在做一个需要旋钮控制的项目用到了EC11旋转编码器。这东西在音响音量调节、菜单选择等场景里很常见但第一次接触时我也被它的A相、B相信号搞晕过。今天我就以立创的GD32E230C8T6开发板为例手把手带你从硬件连接到软件驱动完整实现EC11的旋转方向判断和按键检测重点讲讲怎么处理机械抖动这个“坑”。学完这篇你不仅能看懂EC11的工作原理还能自己写代码让单片机准确识别你是往左拧还是往右拧以及有没有按下旋钮。1. EC11旋转编码器它到底是怎么工作的咱们先别急着接线写代码搞清楚原理后面才不会懵。你可以把EC11想象成一个高级的、带按键的“音量旋钮”。它和普通的电位器用电位变化表示位置完全不同。EC11旋转时会输出两路方波脉冲信号我们叫它们A相和B相有时也标为CLK和DT。核心秘密就在这两路信号的相位关系上。当顺时针旋转时A相信号的变化比如从高电平跳到低电平总是比B相早一点点反过来逆时针旋转时B相的变化就比A相早。这个“早一点点”在波形图上看起来就是两个信号之间有90度的相位差。注意这里的“相位差”是电子信号上的概念不是让你去量角度。你只需要记住判断方向就是看A、B两个引脚谁先变化以及变化时另一个引脚是什么状态。因为是机械结构旋钮的金属触点闭合或断开时不可避免会产生一连串快速的、不稳定的通断这就是抖动。如果不处理单片机可能把你拧一下识别成拧了好几下。所以消抖是驱动EC11必须做的一步。2. 硬件连接把模块接到开发板上EC11模块通常有5个引脚我们用立创GD32E230C8T6开发板来连接它。模块引脚标号引脚名称连接说明连接到GD32E230引脚VCC电源正极开发板3.3V或5V引脚模块支持5V-GND电源地开发板GND引脚SWSW (Switch)按键信号输出任选一个GPIO如PA0配置为上拉输入DTB相旋转信号B相输出任选一个GPIO如PA1配置为上拉输入CLKA相旋转信号A相输出任选一个GPIO如PA2配置为上拉输入接线要点电源模块工作电压是5V但GD32E230的IO口是3.3V电平。模块输出信号在5V供电时是高电平5V不能直接接到3.3V的单片机IO上否则可能损坏芯片稳妥的做法是要么给开发板的VDD供5V电如果MCU支持让IO口输出高电平为5V。要么在A、B、SW信号线上串联一个1kΩ左右的电阻或者使用电平转换电路。最省事的方法是直接用开发板的3.3V给EC11模块供电。大多数EC11模块在3.3V下也能正常工作只是输出高电平变为3.3V这样就能安全连接了。我实测过大部分没问题。上拉电阻EC11模块输出是开漏模式需要外部上拉电阻才能输出高电平。幸运的是GD32E230的GPIO内部可以配置上拉电阻我们直接在软件里开启就行省了外接电阻。引脚选择A、B、SW三个引脚没有特殊功能要求选择普通的GPIO口即可记得在代码里配置好。我的连接方案是开发板3.3V - 模块VCC 开发板GND - 模块GND 模块CLK(A相) - PA2 模块DT(B相) - PA1 模块SW - PA0。3. 软件驱动从引脚配置到方向判断硬件接好了现在来写代码。我们的目标是每隔一段时间检查一次引脚状态根据A、B相的变化规律判断旋转方向并且要能检测按键是否按下。3.1 初始化GPIO首先要把用到的三个引脚PA0, PA1, PA2都初始化为输入模式并且使能内部上拉电阻。这样当编码器没有输出时引脚会被拉成高电平。#include gd32e23x.h void Encoder_GPIO_Init(void) { rcu_periph_clock_enable(RCU_GPIOA); // 使能GPIOA的时钟 /* 配置PA0 (SW按键) 为上拉输入 */ gpio_mode_set(GPIOA, GPIO_MODE_INPUT, GPIO_PUPD_PULLUP, GPIO_PIN_0); /* 配置PA1 (DT/B相) 为上拉输入 */ gpio_mode_set(GPIOA, GPIO_MODE_INPUT, GPIO_PUPD_PULLUP, GPIO_PIN_1); /* 配置PA2 (CLK/A相) 为上拉输入 */ gpio_mode_set(GPIOA, GPIO_MODE_INPUT, GPIO_PUPD_PULLUP, GPIO_PIN_2); }3.2 理解方向判断的真值表这是整个驱动的核心逻辑。我们不是一直盯着引脚看而是每隔一段时间比如10ms去“采样”一下A相和B相的电平。通过比较本次采样和上次采样的值我们能知道引脚有没有发生“跳变”从高变低或从低变高。根据资料里的分析我们可以总结出这个判断逻辑顺时针旋转A相和B相的电平变化方向相同。即如果A相从高变低下降沿那么B相也应该是低电平或也正在变低如果A相从低变高上升沿那么B相也应该是高电平。逆时针旋转A相和B相的电平变化方向相反。即如果A相是下降沿B相却是高电平如果A相是上升沿B相却是低电平。简单记同顺异逆变化相同是顺时针变化相反是逆时针。3.3 定时扫描与消抖实现机械抖动通常持续几毫秒到十几毫秒。我们采用定时器中断扫描法来消抖设置一个定时器每10ms中断一次。在中断里我们只读取一次A、B相的状态并与上一次的状态进行比较判断。由于10ms远大于抖动时间当旋钮稳定在一个位置后我们读到的就是稳定后的正确状态自然就避开了抖动期间的错误信号。这里我们用SysTick定时器系统滴答定时器来实现10ms定时它配置简单。// 定义全局变量用于保存上一次的引脚状态 static uint8_t s_Last_CLK_State 1; // A相上次状态默认高电平上拉 static uint8_t s_Last_DT_State 1; // B相上次状态默认高电平上拉 // 旋转事件计数正数为顺时针负数为逆时针 static int32_t s_Encoder_Counter 0; // 在SysTick中断服务函数中调用配置为10ms中断一次 void Encoder_Scan_Task(void) { uint8_t current_clk gpio_input_bit_get(GPIOA, GPIO_PIN_2); // 读取当前A相(PA2) uint8_t current_dt gpio_input_bit_get(GPIOA, GPIO_PIN_1); // 读取当前B相(PA1) // 检查A相是否发生了变化有跳变 if(current_clk ! s_Last_CLK_State) { // A相发生变化时根据B相的状态判断方向 // 如果A相和B相当前状态相同则是顺时针 if(current_clk current_dt) { s_Encoder_Counter; // 顺时针计数器加 } else // 状态不同则是逆时针 { s_Encoder_Counter--; // 逆时针计数器减 } } // 更新上一次的状态记录 s_Last_CLK_State current_clk; s_Last_DT_State current_dt; } // 获取旋转计数值的函数 int32_t Encoder_Get_Count(void) { int32_t count; // 读取时暂时关闭中断防止在读取过程中被中断修改保证数据完整性 __disable_irq(); count s_Encoder_Counter; __enable_irq(); return count; } // 按键检测函数按键按下时SW引脚会变为低电平 uint8_t Encoder_Sw_Down(void) { // 读取SW引脚(PA0)按下为0松开为1因为上拉了 if(gpio_input_bit_get(GPIOA, GPIO_PIN_0) 0) { delay_ms(20); // 简单延时消抖 if(gpio_input_bit_get(GPIOA, GPIO_PIN_0) 0) { return 1; // 确认按下 } } return 0; // 未按下 }代码关键点解释Encoder_Scan_Task函数每10ms执行一次它对比A相当前状态和上次保存的状态如果不同说明A相有跳变。一旦检测到A相跳变立刻查看此时B相的电平。根据“同顺异逆”规则判断方向并修改计数器s_Encoder_Counter。按键检测Encoder_Sw_Down用了简单的延时消抖第一次检测到低电平后延时20ms再检测一次如果还是低电平就确认是真正的按下。3.4 主程序与现象验证把上面的函数整合到你的工程里。记得初始化系统时钟、SysTick定时器配置为10ms中断并在中断里调用Encoder_Scan_Task和串口用于打印信息。#include gd32e23x.h #include systick.h #include bsp_usart.h #include stdio.h // 声明外部函数 extern void Encoder_GPIO_Init(void); extern int32_t Encoder_Get_Count(void); extern uint8_t Encoder_Sw_Down(void); int main(void) { systick_config(); // 初始化SysTick定时器配置为1ms中断需调整中断周期为10ms usart_gpio_config(115200U); // 初始化串口波特率115200 Encoder_GPIO_Init(); // 初始化编码器GPIO printf(EC11 Encoder Demo Start\r\n); int32_t last_count 0; while(1) { // 检测旋转 int32_t current_count Encoder_Get_Count(); if(current_count ! last_count) { printf(Encoder Count: %ld\r\n, current_count); last_count current_count; } // 检测按键 if(Encoder_Sw_Down() 1) { printf(Encoder Button Pressed!\r\n); delay_ms(300); // 按键去抖防止连续打印 } delay_ms(50); // 主循环延时降低CPU占用 } }编译下载后你可以在串口助手上看到顺时针旋转编码器打印的Encoder Count数值会不断增加。逆时针旋转数值会不断减少。按下编码器的轴按键会打印Encoder Button Pressed!。4. 常见问题与调试心得方向判断反了如果发现顺时针打印的数值在减少逆时针在增加说明你的A、B相引脚接反了或者代码里“同顺异逆”的逻辑写反了。最简单的解决办法是交换一下开发板上A相和B相的接线。计数不准确偶尔跳变这很可能是消抖时间不够。尝试把定时扫描间隔从10ms加大到15ms或20ms试试。另一个可能是电源噪声确保电源稳定并在VCC和GND之间加一个0.1uF的滤波电容。按键反应不灵或连发调整Encoder_Sw_Down函数里的延时消抖时间。20ms是个常用起点如果环境干扰大可以加到50ms。如果按下一次串口打印多次说明松手检测也有抖动可以在检测到按键释放时也加个延时判断。旋转很慢时计数丢失我们目前的算法是“跳变检测”只在状态变化时计数。如果旋转得非常慢10ms的扫描周期可能捕捉不到一次完整的跳变。对于低速高精度应用可以考虑使用“状态机”或“四倍频”算法它能在一个周期内检测到4次跳变精度更高但代码也稍复杂。这个基于定时器扫描的EC11驱动方法在大多数项目中已经足够稳定可靠了。关键是理解了相位差判断方向的原理以及用定时中断避开机械抖动的思路。你可以把这个驱动代码当作一个模块轻松移植到其他需要旋钮控制的项目中去。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2415350.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!