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

STM32矩阵按键详解——4×4行列扫描与非阻塞消抖(硬件总结六)

前言独立按键虽然简单但当产品需要十几个按键时每个按键独占一个GPIO的接法就变得很不经济。矩阵按键通过“行×列”的交叉结构仅用NM个GPIO即可驱动N×M个按键。以最常见的4×4矩阵为例16个按键仅需8个GPIO引脚利用率提升整整一倍。本文将从硬件电路出发深入讲解行列扫描算法给出完整的标准库驱动代码并实现非阻塞消抖和可靠的事件标记机制。所有代码基于STM32F103C8T6可直接在工程中编译运行。一、矩阵按键的硬件结构1.1 物理连接矩阵按键由行线Row与列线Column交叉构成。每个按键位于某一行线与某一列线的交点处按下时使该行与该列导通。C0 C1 C2 C3 │ │ │ │ R0 ──┼────┼────┼────┼── ╹ ╹ ╹ ╹ R1 ──┼────┼────┼────┼── ╹ ╹ ╹ ╹ R2 ──┼────┼────┼────┼── ╹ ╹ ╹ ╹ R3 ──┼────┼────┼────┼──行线R0~R3配置为推挽输出扫描时依次拉低。列线C0~C3配置为上拉输入内部上拉或外部上拉电阻默认读高电平。当某一行被拉低、该行和某列交叉点的按键闭合时列线通过闭合触点被拉低程序即可检测到低电平。1.2 上拉电阻与引脚配置列线必须接上拉电阻以保证悬空时读到确定的高电平。推荐使用外部10kΩ上拉电阻。STM32也可直接配置为GPIO_Mode_IPU利用内部约40kΩ弱上拉但抗干扰能力较弱。本文引脚分配如下使用PA0~PA7功能引脚说明行0PA0推挽输出行1PA1推挽输出行2PA2推挽输出行3PA3推挽输出列0PA4上拉输入列1PA5上拉输入列2PA6上拉输入列3PA7上拉输入1.3 安全注意事项任何时刻只能拉低一行其余行必须输出高电平。如果同时有两行分别输出高和低当同一列上的两个不同行按键同时按下时高电平的行将与低电平的行发生短路可能损坏GPIO。这是软件必须保证的约束。推挽输出只要遵守此规则就完全安全。1.4 幽灵键问题Ghost Key多键同时按下时电流可能通过已闭合的触点形成反向通路导致未按下的按键被误判为按下。对于常规应用可在软件中检测到多于2个键同时按下时直接丢弃本次扫描结果若要求绝对可靠需在每个按键上串联二极管如1N4148。二、行列扫描算法2.1 基本流程将所有行线置高电平。逐行扫描依次将每一行拉低其余行保持高电平同时读取所有列线的状态。若某列读到低电平说明被拉低的这一行与该列的交叉点上的按键被按下。扫描完所有行后综合结果可获知全部被按下的按键。2.2 消抖策略机械按键存在5~20ms的抖动。我们采用固定周期扫描状态机消抖通过SysTick产生10ms定时在MatrixKey_Scan()中自动限制扫描间隔。为每个按键维护一个消抖计数器只有连续两次扫描检测到电平与当前稳定状态不同时才更新稳定状态。稳定状态变化时产生“按下”或“释放”事件并标记待消费。三、标准库完整实现可直接使用3.1 头文件与宏#includestm32f10x.h#includestdbool.h/* 引脚定义 */#defineKEY_PORTGPIOA#defineKEY_ROW0_PINGPIO_Pin_0#defineKEY_ROW1_PINGPIO_Pin_1#defineKEY_ROW2_PINGPIO_Pin_2#defineKEY_ROW3_PINGPIO_Pin_3#defineKEY_COL0_PINGPIO_Pin_4#defineKEY_COL1_PINGPIO_Pin_5#defineKEY_COL2_PINGPIO_Pin_6#defineKEY_COL3_PINGPIO_Pin_7#defineKEY_ROWS4#defineKEY_COLS4#defineKEY_NUM(KEY_ROWS*KEY_COLS)/* 16 *//* 消抖参数 */#defineDEBOUNCE_MS20#defineSCAN_INTERVAL_MS10// 扫描间隔10ms消抖需2次确认/* LED */#defineLED_GPIOGPIOB#defineLED_PINGPIO_Pin_03.2 按键状态结构typedefenum{KEY_STATE_IDLE0,KEY_STATE_PRESS,KEY_STATE_RELEASE}KeyState;typedefstruct{uint8_tdebounce_cnt;// 消抖计数bool current_raw;// 当前原始电平true未按下bool stable;// 消抖后的稳定状态true未按下KeyState state;// 按键状态bool event_consumed;// 事件是否已被消费}KeyInfo;staticKeyInfo key_info[KEY_NUM];/* 字符映射表 */staticconstcharkey_map[KEY_ROWS][KEY_COLS]{{1,2,3,A},{4,5,6,B},{7,8,9,C},{*,0,#,D}};3.3 时基SysTickvolatileuint32_tsysTickUptime0;voidSysTick_Init(void){if(SysTick_Config(SystemCoreClock/1000)){while(1);}NVIC_SetPriority(SysTick_IRQn,0x0F);}voidSysTick_Handler(void){sysTickUptime;}3.4 GPIO初始化voidMatrixKey_GPIO_Init(void){GPIO_InitTypeDef GPIO_InitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);/* 行线 PA0~PA3 推挽输出初始全高 */GPIO_InitStructure.GPIO_PinKEY_ROW0_PIN|KEY_ROW1_PIN|KEY_ROW2_PIN|KEY_ROW3_PIN;GPIO_InitStructure.GPIO_ModeGPIO_Mode_Out_PP;GPIO_InitStructure.GPIO_SpeedGPIO_Speed_50MHz;GPIO_Init(KEY_PORT,GPIO_InitStructure);GPIO_SetBits(KEY_PORT,KEY_ROW0_PIN|KEY_ROW1_PIN|KEY_ROW2_PIN|KEY_ROW3_PIN);/* 列线 PA4~PA7 上拉输入 */GPIO_InitStructure.GPIO_PinKEY_COL0_PIN|KEY_COL1_PIN|KEY_COL2_PIN|KEY_COL3_PIN;GPIO_InitStructure.GPIO_ModeGPIO_Mode_IPU;// 内部上拉GPIO_Init(KEY_PORT,GPIO_InitStructure);}/** * brief 初始化按键状态数组确保上电时状态为“未按下” */voidMatrixKey_State_Init(void){for(uint8_ti0;iKEY_NUM;i){key_info[i].stabletrue;// 初始化为未按下key_info[i].stateKEY_STATE_IDLE;key_info[i].event_consumedtrue;key_info[i].debounce_cnt0;}}3.5 底层扫描函数/** * brief 读取指定行列按键的原始电平 * param row 行号 (0~3), col 列号 (0~3) * retval true: 未按下高电平 false: 按下低电平 */staticboolMatrixKey_ReadRaw(uint8_trow,uint8_tcol){constuint16_trow_pin[KEY_ROWS]{KEY_ROW0_PIN,KEY_ROW1_PIN,KEY_ROW2_PIN,KEY_ROW3_PIN};constuint16_tcol_pin[KEY_COLS]{KEY_COL0_PIN,KEY_COL1_PIN,KEY_COL2_PIN,KEY_COL3_PIN};/* 全部行先拉高再拉低目标行 */GPIO_SetBits(KEY_PORT,KEY_ROW0_PIN|KEY_ROW1_PIN|KEY_ROW2_PIN|KEY_ROW3_PIN);GPIO_ResetBits(KEY_PORT,row_pin[row]);/* 极短延时等待电平稳定 */for(volatileuint8_td0;d5;d);/* 读取列状态 */return(GPIO_ReadInputDataBit(KEY_PORT,col_pin[col])!Bit_RESET);}3.6 消抖与扫描状态机/** * brief 矩阵按键扫描函数每10ms调用一次 * 内部完成消抖和状态迁移为每个按键产生一次性事件 */voidMatrixKey_Scan(void){staticuint32_tlast_scan0;if(sysTickUptime-last_scanSCAN_INTERVAL_MS)return;last_scansysTickUptime;for(uint8_trow0;rowKEY_ROWS;row){for(uint8_tcol0;colKEY_COLS;col){uint8_tidxrow*KEY_COLScol;KeyInfo*kkey_info[idx];k-current_rawMatrixKey_ReadRaw(row,col);/* 消抖计数器与稳定状态不同则累加相同则清零 */if(k-current_rawk-stable){k-debounce_cnt0;}else{k-debounce_cnt;if(k-debounce_cnt(DEBOUNCE_MS/SCAN_INTERVAL_MS)){// 电平连续2次(20ms)与当前stable不同更新stablek-stablek-current_raw;k-debounce_cnt0;if(k-stablefalse){/* 确认按下 */if(k-state!KEY_STATE_PRESS){k-stateKEY_STATE_PRESS;k-event_consumedfalse;// 新事件待消费}}else{/* 确认释放 */k-stateKEY_STATE_RELEASE;k-event_consumedfalse;}}}}}}3.7 应用层API/** * brief 检测指定按键是否刚被按下一次性事件调用后即清除 * param row, col 按键位置 * retval true: 有新的按下事件 false: 无 */boolMatrixKey_IsPressed(uint8_trow,uint8_tcol){uint8_tidxrow*KEY_COLScol;KeyInfo*kkey_info[idx];if(k-stateKEY_STATE_PRESS!k-event_consumed){k-event_consumedtrue;returntrue;}returnfalse;}/** * brief 检查指定按键是否处于按住状态可用于长按连发 * retval true: 按键处于按下状态 false: 未按下 */boolMatrixKey_IsDown(uint8_trow,uint8_tcol){uint8_tidxrow*KEY_COLScol;return(key_info[idx].stateKEY_STATE_PRESS);}/** * brief 获取按键对应的字符 */charMatrixKey_GetChar(uint8_trow,uint8_tcol){returnkey_map[row][col];}3.8 主函数示例intmain(void){GPIO_InitTypeDef GPIO_InitStructure;/* LED PB0 推挽输出 */RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);GPIO_InitStructure.GPIO_PinLED_PIN;GPIO_InitStructure.GPIO_ModeGPIO_Mode_Out_PP;GPIO_InitStructure.GPIO_SpeedGPIO_Speed_50MHz;GPIO_Init(LED_GPIO,GPIO_InitStructure);GPIO_ResetBits(LED_GPIO,LED_PIN);// 初始熄灭/* 矩阵按键初始化 */MatrixKey_GPIO_Init();MatrixKey_State_Init();// 关键清空状态SysTick_Init();while(1){MatrixKey_Scan();// 每循环都调用内部自动限速/* 处理所有按键事件 */for(uint8_tr0;rKEY_ROWS;r){for(uint8_tc0;cKEY_COLS;c){if(MatrixKey_IsPressed(r,c)){charchMatrixKey_GetChar(r,c);/* 在此处理按键事件例如翻转LED */GPIO_WriteBit(LED_GPIO,LED_PIN,(BitAction)(1-GPIO_ReadOutputDataBit(LED_GPIO,LED_PIN)));// 也可通过串口打印: printf(Key: %c\r\n, ch);}}}/* 示例检查“*”键是否按住连续动作 */if(MatrixKey_IsDown(3,0)){// 第3行第0列即*// 执行连续操作如持续调亮度}}}代码说明MatrixKey_State_Init()将全部按键的稳定状态初始化为未按下防止上电误触发。MatrixKey_Scan()内部由sysTickUptime控制10ms间隔即使主循环调用再快也不会频繁扫描。每个按键的event_consumed保证一次按下只产生一次IsPressed事件长按期间不会重复触发。IsDown()提供持续按住的状态可用于实现长按加速等逻辑。四、扩展建议长按识别可在每个按键上增加按下时间戳当IsDown()为真且持续时间超过阈值时触发长按事件需自行扩展状态机。组合键同时检查多个按键的IsDown()状态即可。低功耗将MatrixKey_Scan()放入定时中断主循环空闲时调用__WFI()可大幅降低功耗。五、常见问题排查现象可能原因解决方法按键无反应行线未输出、列线上拉未使能检查GPIO_Mode_Out_PP和GPIO_Mode_IPU单次按下触发多次事件未消费、消抖不足确认event_consumed机制检查扫描间隔多键同时按下误判幽灵键效应软件丢弃2键同时按下的结果或硬件加二极管上电后自动触发一次初始状态未校准调用MatrixKey_State_Init()按键响应慢扫描间隔太长减小SCAN_INTERVAL_MS建议10ms六、总结本文从矩阵按键的硬件原理出发深入讲解了行列扫描算法并给出了一套完整、可直接使用的标准库驱动。通过固定周期扫描配合消抖状态机实现了精准、非阻塞的按键识别且事件消费机制严谨可靠。这套代码与之前文章中的状态机、调度器及低功耗方案完全兼容稍作整合即可构建出复杂而稳定的裸机交互系统。若有任何疑问欢迎在评论区留言交流

相关文章:

STM32矩阵按键详解——4×4行列扫描与非阻塞消抖(硬件总结六)

前言 独立按键虽然简单,但当产品需要十几个按键时,每个按键独占一个GPIO的接法就变得很不经济。矩阵按键通过“行列”的交叉结构,仅用NM个GPIO即可驱动NM个按键。以最常见的44矩阵为例,16个按键仅需8个GPIO,引脚利用率…...

鸿蒙中的自由流转

鸿蒙自由流转是 ‌HarmonyOS(鸿蒙系统)‌ 实现多设备协同的核心能力之一,旨在打破设备边界,让应用和服务在不同终端间无缝流转,提升用户体验。‌什么是鸿蒙自由流转?‌鸿蒙自由流转是指用户在多个搭载 Harm…...

RUST编程学习.2语法

目录 前言 一、思维导图 二、Rust语法专属 1.迭代器 2.生命周期 总结 前言 在进行编译器下载后,就可以写代码进行编译调试了,在这之前就是要学习rust语法,在学习的过程中我整理了一版思维导图,最直观的感觉就是rust的语法很…...

【SSD】闪存1

闪存的特点 闪存是非易失存储器,掉电了数据也不会丢失,但是闪存不能够覆写,必须按块擦除,按页写入。 闪存的基本单元 闪存的基本单元是Cell,一种类Nmos的双层浮栅MOS管 MOS管 首先理解什么是MOS管:(金…...

性价比高的国产PLM软件公司

在制造业领域,不少企业都面临着研发效率低下、协同困难等问题。比如某电子制造企业,研发部门与生产部门之间信息沟通不畅,图纸版本管理混乱,导致产品研发周期延长,生产成本增加,新品上市时间比预期晚了近30…...

分布式团队的代码协作规范:从分支策略到提交信息格式

在分布式团队模式下,代码协作的地域分散、时区差异和沟通成本,给版本控制和质量保障带来了严峻挑战。作为软件测试从业者,我们不仅是代码质量的“守门员”,更需要深入理解并推动执行规范的代码协作流程,从分支管理到提…...

几十万买的数字孪生低代码平台集体落灰?被隐瞒的落地真相,终于说透了

在政企数字化采购圈子里,一直有个特别讽刺、且年年重复上演的现象。很多企业、政府单位,手握专项数字化预算,毫不犹豫花几十万重金购入数字孪生、3D可视化低代码平台。采购前被厂商的宣传话术打动:零代码拖拽、人人上手、无需专业…...

在家办公效率低?试试这个“空间切换”技巧

一、软件测试从业者居家办公的效率困境对于软件测试从业者而言,居家办公看似摆脱了办公室的嘈杂与束缚,实则面临着诸多独特的效率挑战。测试工作本身就需要高度的专注与严谨,从需求分析、用例设计到缺陷跟踪,每一个环节都容不得半…...

SMARTFORM不同模板一起打印

一、背景由于客户提出发货单要加上条形码打印,条形码单独一个模板,加在后面打印,输出PDF并发送邮件。二、效果展示不同模板一起打印效果如下,建立smartforms的表单时,也使用了两个不同的模板三、smartforms建立表单&am…...

远程技术面试的潜规则:摄像头角度可能影响你的录用

一、摄像头角度:被忽视的专业细节在软件测试的工作中,我们习惯用严谨的态度去排查代码里的每一个bug,用精准的测试用例去验证产品的每一项功能。但在远程技术面试这个特殊的“测试场景”里,很多人却忽略了一个看似无关紧要&#x…...

Spec-Kit + Superpowers 实战:Go语言博客论坛系统的规范驱动开发

从“凭感觉写代码”到“按规范做工程”,一套完整的AI驱动开发方法论落地 一、引言:AI编程的“效率陷阱” 2024年Google DORA报告揭示了一个令人困惑的数据:AI编码助手采用率每提升25%,软件交付稳定性反而下降7.2%。问题出在哪?研究表明,当上下文从1K Token扩展到32K Tok…...

微波遥感杂谈五(微波辐射计)

前言微波辐射计是通过被动的接收各个高度传来的温度辐射的微波信号来判断温度、 湿度曲线,能定量测量目标(如地物和大气各成分)的低电平微波辐射的高灵敏度接收装置。目前机载微波辐射计实测温度分辨率达0.02K,星载微波辐射计温度分辨率达 0.2&#xff5…...

适配多层级组织管理,科学运用 360 度反馈打造公平高效绩效文化

360度绩效反馈评估是一种从上级、下属、同事、客户等多个维度收集反馈的综合绩效评估方法,通过多源数据消除单一评价者的主观偏差,帮助企业获得更全面、客观的员工能力画像。相比传统的上级单向评价,360度反馈能将评估准确度提升40%以上&…...

鸿蒙备考题库页面构建:错题本、小组榜单与备考提示模块详解

鸿蒙备考题库页面构建:错题本、小组榜单与备考提示模块详解 前言 在 HarmonyOS 6.0 应用开发中,教育类应用的错题管理、学习排行榜和系统提示是提升用户粘性的关键功能模块。本文将以“备考题库”应用中的“错题本”高频错题列表、“小组榜单”学习排名和…...

中文编程语言的开创性语法,言律:一门以汉语为思维内核的原生中文编程语言

在对母语作为思维原生载体的深层结构、语言相对论与神经认知机制的探讨基础上,我们不再满足于“把英文关键字翻译成中文”的表层汉化,而是要开创一种真正根植于汉语思维逻辑的编程语法体系—— 🌿「言律」(Yn Lǜ)&…...

leetcode思路-236 二叉树的最近公共祖先

给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖先的定义为:“对于有根树 T 的两个节点 p、q,最近公共祖先表示为一个节点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节点也可以是它自己的…...

把SAC model的数据导出到BW的ADSO中

目录 1. SAC 侧的准备 1.1 OData连接要做好 1.2 SAC里的model设置要配置好允许导出到Odata 2. BW侧要做的准备(先跟着SAP的note走) 3. SAC 模型数据导出 一般都是把planning model的数据导出到一个ADSO中,然后再用Composite Provider里…...

Word文档保护技巧:防止内容被轻易复制

Word文档如何防止复制呢?其实,Word根本没有真正意义上的禁止复制,因为用户按一下手机截图,或者拍张照片,内容照样能拿走。但是,我们可以提高复制门槛,也就是让其他用户通过“CtrlC”无法直接复制…...

计算机毕业设计 | SpringBoot+vue医院药品管理系统(附源码+论文)

1,绪论 1.1课题背景 身处网络时代,随着网络系统体系发展的不断成熟和完善,人们的生活也随之发生了很大的变化。目前,人们在追求较高物质生活的同时,也在想着如何使自身的精神内涵得到提升,而读书就是人们…...

机器学习之逻辑回归算法

一、逻辑回归简介 1. 定义 逻辑回归(Logistic Regression)是一种有监督学习算法,主要用于解决二分类问题的统计学习方法。尽管名字中带有“回归”,但它实际上是一种分类算法。 大白话解释 逻辑回归就是一种“做判断题”的算法&…...

强制启动 Cursor IDE 主程序(不带 Agent 模式)

🔧 终极解决:强制启动 Cursor IDE 主程序(不带 Agent 模式)方法 1:用「命令行」强制启动主程序(最稳)按 WinR 打开运行窗口,输入 cmd 回车,打开命令提示符输入下面这行命…...

Ollama访问限制

发布于: Ollama访问限制 | Eucalyptushttps://blog.mingliangstar.com/2026/05/21/Ollama%E8%AE%BF%E9%97%AE%E9%99%90%E5%88%B6/ NginxBasic Auth认证 生成密码文件 # 安装工具 yum install httpd-tools -y# 创建密码文件(用户名 admin)…...

7.跨品牌手机刷机原理深度解析|BL 解锁机制 + 分区读写 + 故障修复全方案

摘要 本文系统性地阐述主流品牌智能手机(华为、小米、OPPO、vivo、一加、苹果)刷机与维修的核心原理与操作流程。针对不同品牌底层架构差异,提供从Bootloader解锁、Recovery刷写到系统固件注入的完整技术方案。所有操作步骤均基于实际硬件环境验证,包含完整可运行的Python…...

【K8s】解惑:K8s 与 Docker 的关系

目录 引言:一个绕不开的问题 一句话说清K8s与Docker的关系 澄清三个误解 从命令的角度,直观对比 引言:一个绕不开的问题 在学习云原生技术的路上,几乎每个人都会遇到这样一个困惑: “有了 Kubernetes&#xff08…...

6.解决 99% 刷机故障|GPT 分区修复 + SEP 兼容检测 + 全分区备份,工程师实战手册

摘要 本文面向具备基础Linux命令行操作能力的维修工程师与高级发烧友,系统阐述主流品牌手机刷机与维修的底层逻辑与标准化操作流程。内容覆盖高通、联发科、苹果A系列三大芯片平台的刷机协议差异,提供完整的刷机工具链搭建脚本、分区备份恢复脚本、以及底层驱动级故障诊断代…...

论文AI率爆表怕延毕?5招实测降AI率,3分钟知网AIGC过审上岸

2025 年 12 月 25 日知网 AIGC 检测系统升级,2026 年 4 月 27 日维普 AI 率检测平台升级…2026 毕业季,各大主流 AIGC 检测软件陆续升级系统,识别 AI 痕迹更加精准。 临近毕业,同学们看者飘红的 AIGC 检测报告、纷繁复杂的降 AI …...

探索C/C++程序从源码到可执行文件的编译之旅

计算机程序设计语言可大致分为机器语言、汇编语言和高级语言三类。高级语言需经翻译成机器语言后方可执行,此翻译过程包括编译和解释两种方式。因此,高级语言可进一步细分为编译型语言和解释型语言。编译型语言如C、C、Java,其程序需经过预处…...

AI犯了错没人追责,工程师犯了错丢饭碗?

芯片公司开始大量引入AI辅助设计工具,生成RTL代码、跑仿真、做时序分析。与此同时,公司对工程师的容错空间越来越小,考核越来越严,出了bug第一反应是找人背锅。这两件事放在一起,细想一下,其实挺荒诞的。AI…...

通宵降AI率?10款降AI工具亲测:哪个神器一次过,哪个白花钱

2025 年 12 月 25 日知网 AIGC 检测系统升级,2026 年 4 月 27 日维普 AI 率检测平台升级…2026 毕业季,各大主流 AIGC 检测软件陆续升级系统,识别 AI 痕迹更加精准。 临近毕业,同学们看者飘红的 AIGC 检测报告、纷繁复杂的降 AI 系…...

忙碌”幻觉:你以为在推进项目,其实只是在逃避

时序收敛没过、功耗超了、验证卡住了——每一个问题都是真实的,每一项任务都是紧迫的。但有时候停下来想想,这些忙碌背后,到底有多少是真正在解决问题,有多少只是在用”我还在干活”这件事本身,来麻醉自己?…...