按键状态机

原工程地址:https://github.com/candylife9/state_machine_example

视频:C语言之状态机编程_02_状态机使用案例分析_哔哩哔哩_bilibili

我觉得讲的挺好的。

来自豆包封装的通用接口

头文件

/*** @file key_state_machine.h* @brief 通用按键状态机接口*/#ifndef KEY_STATE_MACHINE_H
#define KEY_STATE_MACHINE_H#include <stdint.h>
#include <stdbool.h>/** 按键状态枚举 */
typedef enum {KEY_STATE_IDLE,           // 空闲状态KEY_STATE_PRESS_DEBOUNCE, // 按下消抖状态KEY_STATE_SHORT_PRESS,    // 短按状态KEY_STATE_LONG_PRESS,     // 长按状态KEY_STATE_RELEASE_DEBOUNCE // 释放消抖状态
} KeyState;/** 按键事件回调函数类型 */
typedef void (*KeyEventCallback)(void* context, uint8_t key_id, bool is_long_press);/** 按键配置结构体 */
typedef struct {uint8_t key_id;                  // 按键IDuint32_t debounce_time_ms;       // 消抖时间(ms)uint32_t long_press_threshold_ms; // 长按阈值(ms)uint32_t long_press_interval_ms;  // 长按连续触发间隔(ms)bool active_low;                 // 是否低电平有效bool (*read_pin)(void);          // 读取引脚电平的函数指针KeyEventCallback event_callback; // 事件回调函数void* callback_context;          // 回调函数上下文
} KeyConfig;/** 按键状态机实例结构体 */
typedef struct {const KeyConfig* config;         // 按键配置KeyState state;                  // 当前状态uint8_t debounce_counter;        // 消抖计数器uint32_t check_time;             // 检测时间戳uint32_t old_time;               // 旧时间戳bool press_detected;             // 短按检测标志bool long_press_detected;        // 长按检测标志uint32_t long_press_trigger_time; // 长按触发时间
} KeyStateMachine;/*** @brief 初始化按键状态机* @param key 按键状态机实例指针* @param config 按键配置指针*/
void Key_Init(KeyStateMachine* key, const KeyConfig* config);/*** @brief 按键状态机处理函数,需定期调用* @param key 按键状态机实例指针*/
void Key_Process(KeyStateMachine* key);/*** @brief 获取按键当前状态* @param key 按键状态机实例指针* @return 当前状态*/
KeyState Key_GetState(const KeyStateMachine* key);/*** @brief 检查按键是否被按下(短按或长按)* @param key 按键状态机实例指针* @return true: 按下, false: 未按下*/
bool Key_IsPressed(const KeyStateMachine* key);/*** @brief 检查按键是否处于长按状态* @param key 按键状态机实例指针* @return true: 长按, false: 非长按*/
bool Key_IsLongPressed(const KeyStateMachine* key);#endif // KEY_STATE_MACHINE_H

C文件 

/*** @file key_state_machine.c* @brief 通用按键状态机实现*/#include "key_state_machine.h"
#include "stm32fxxx_hal.h" // 假设使用STM32系列,需要包含HAL库头文件/*** @brief 判断引脚电平是否为有效电平* @param key 按键状态机实例指针* @return true: 有效电平, false: 无效电平*/
static bool IsKeyLevelActive(const KeyStateMachine* key) {bool pin_state = key->config->read_pin();return (key->config->active_low) ? (pin_state == false) : (pin_state == true);
}void Key_Init(KeyStateMachine* key, const KeyConfig* config) {if (!key || !config) return;key->config = config;key->state = KEY_STATE_IDLE;key->debounce_counter = 0;key->check_time = HAL_GetTick();key->old_time = HAL_GetTick();key->press_detected = false;key->long_press_detected = false;key->long_press_trigger_time = 0;
}void Key_Process(KeyStateMachine* key) {if (!key || !key->config) return;uint32_t current_time = HAL_GetTick();switch (key->state) {case KEY_STATE_IDLE:if (current_time - key->old_time > 1000) {// 空闲时,每间隔1秒输出等待按键提示信息(可通过回调函数实现)if (key->config->event_callback) {key->config->event_callback(key->config->callback_context, key->config->key_id, false);}key->old_time = current_time;}if (IsKeyLevelActive(key)) {// 检测到有效电平,开始消抖key->check_time = current_time;key->debounce_counter = 5; // 默认5次检测key->state = KEY_STATE_PRESS_DEBOUNCE;}break;case KEY_STATE_PRESS_DEBOUNCE:if (key->debounce_counter > 0) {if (current_time - key->check_time > key->config->debounce_time_ms) {if (!IsKeyLevelActive(key)) {// 检测到无效电平,抖动,返回空闲状态key->old_time = current_time;key->state = KEY_STATE_IDLE;} else {// 仍然是有效电平,继续消抖key->check_time = current_time;key->debounce_counter--;}}} else {// 消抖完成,确认按下key->press_detected = true;key->old_time = current_time;key->long_press_trigger_time = current_time;key->state = KEY_STATE_SHORT_PRESS;// 触发短按事件回调if (key->config->event_callback) {key->config->event_callback(key->config->callback_context, key->config->key_id, false);}}break;case KEY_STATE_SHORT_PRESS:key->press_detected = false; // 重置短按标志if (current_time - key->old_time > key->config->long_press_threshold_ms) {// 达到长按阈值,进入长按状态key->long_press_detected = true;key->old_time = current_time;key->state = KEY_STATE_LONG_PRESS;// 触发长按事件回调if (key->config->event_callback) {key->config->event_callback(key->config->callback_context, key->config->key_id, true);}} else if (!IsKeyLevelActive(key)) {// 短按状态下检测到释放,开始释放消抖key->check_time = current_time;key->debounce_counter = 5;key->state = KEY_STATE_RELEASE_DEBOUNCE;}break;case KEY_STATE_LONG_PRESS:if (current_time - key->old_time > key->config->long_press_interval_ms) {// 长按连续触发key->old_time = current_time;// 触发长按连续事件回调if (key->config->event_callback) {key->config->event_callback(key->config->callback_context, key->config->key_id, true);}}if (!IsKeyLevelActive(key)) {// 长按状态下检测到释放,开始释放消抖key->check_time = current_time;key->debounce_counter = 5;key->state = KEY_STATE_RELEASE_DEBOUNCE;}break;case KEY_STATE_RELEASE_DEBOUNCE:if (key->debounce_counter > 0) {if (current_time - key->check_time > key->config->debounce_time_ms) {if (IsKeyLevelActive(key)) {// 检测到有效电平,抖动,返回之前的状态key->state = (key->long_press_detected) ? KEY_STATE_LONG_PRESS : KEY_STATE_SHORT_PRESS;} else {// 仍然是无效电平,继续消抖key->check_time = current_time;key->debounce_counter--;}}} else {// 消抖完成,确认释放key->long_press_detected = false;key->old_time = current_time;key->state = KEY_STATE_IDLE;}break;default:key->state = KEY_STATE_IDLE;break;}
}KeyState Key_GetState(const KeyStateMachine* key) {return (key) ? key->state : KEY_STATE_IDLE;
}bool Key_IsPressed(const KeyStateMachine* key) {if (!key) return false;KeyState state = key->state;return (state == KEY_STATE_SHORT_PRESS || state == KEY_STATE_LONG_PRESS);
}bool Key_IsLongPressed(const KeyStateMachine* key) {return (key && key->state == KEY_STATE_LONG_PRESS);
}

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

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

相关文章

华为OD机试真题——新学校选址(2025A卷:100分)Java/python/JavaScript/C/C++/GO最佳实现

2025 A卷 100分 题型 本专栏内全部题目均提供Java、python、JavaScript、C、C++、GO六种语言的最佳实现方式; 并且每种语言均涵盖详细的问题分析、解题思路、代码实现、代码详解、3个测试用例以及综合分析; 本文收录于专栏:《2025华为OD真题目录+全流程解析+备考攻略+经验分…

欧拉操作系统下安装hadoop集群

背景&#xff1a;欧拉操作系统下安装CDH集群的时候&#xff0c;需要安装python2.7.5&#xff0c;但是本身欧拉系统对python2的支持可能没有那么好&#xff0c;所以考虑搭建原生的hadoop集群。 基础环境如下 组件名称组件版本欧拉VERSION“22.03 (LTS-SP4)”jdkopenjdk versio…

SQL语句的执行流程

文章目录 一、执行流程二、建立连接三、预处理器四、解析器4.1 词法分析4.2 语法分析4.3 语义分析 五、优化器六、执行器七、返回结果 一、执行流程 阶段主要功能关键组件1. 建立连接身份验证、权限检查连接器2. 预处理器缓存检查、SQL预处理查询缓存3. 解析器词法分析、语法分…

TiDB:从快速上手到核心原理与最佳实践

文章目录 引言第一部分&#xff1a;TiDB快速体验与实践指南1. TiDB概述2. TiDB部署方式2.1 本地测试环境部署2.2 生产环境部署2.3 Kubernetes部署2.4 云服务 3. TiDB基本操作3.1 连接TiDB3.2 数据库和表操作3.3 分区表3.4 事务操作 4. 数据迁移到TiDB4.1 从MySQL迁移4.2 使用Ti…

总结:进程和线程的联系和区别

前言:通过学习javaEE初阶中的多线程章节后加上我自己的理解,想来总结一下线程和进程的联系和区别. 一来是能更好地复习知识,二来是为了记录我的学习路程,相信未来的我回首不会忘记这段难忘的经历. 1.进程 先来谈谈进程:进程是操作系统中资源分配的基本单位. 1)进程的执行方…

边缘云的定义、实现与典型应用场景!与传统云计算的区别!

一、什么是边缘云&#xff1f;‌ 边缘云是一种‌分布式云计算架构‌&#xff0c;将计算、存储和网络资源部署在‌靠近数据源或终端用户的网络边缘侧‌&#xff08;如基站、本地数据中心或终端设备附近&#xff09;&#xff0c;而非传统的集中式云端数据中心。 ‌核心特征‌&…

海康威视摄像头C#开发指南:从SDK对接到安全增强与高并发优化

一、海康威视SDK核心对接流程​​ 1. ​​开发环境准备​​ ​​官方SDK获取​​&#xff1a;从海康开放平台下载最新版SDK&#xff08;如HCNetSDK.dll、PlayCtrl.dll&#xff09;。​​依赖项安装​​&#xff1a;确保C运行库&#xff08;如vcredist_x86.exe&#xff09;与S…

《软件工程》第 9 章 - 软件详细设计

目录 9.1 详细设计的任务与过程模型 9.2 用例设计 9.2.1 设计用例实现方案 9.2.2 构造设计类图 9.2.3 整合并优化用例实现方案 9.3 子系统设计 9.3.1 确立内部设计元素 9.3.2 导出设计类图 9.4 构件设计 9.5 类设计 9.5.1 精化类间关系 9.5.2 精化属性和操作 9.5.…

spring+tomcat 用户每次发请求,tomcat 站在线程的角度是如何处理用户请求的,spinrg的bean 是共享的吗

对于 springtomcat 用户每次发请求&#xff0c;tomcat 站在线程的角度是如何处理的 比如 bio nio apr 等情况 tomcat 配置文件中 maxThreads 的数量是相对于谁来说的&#xff1f; 以及 spring Controller 中的全局变量:各种bean 对于线程来说是共享的吗&#xff1f; 一、Tomca…

存储引擎系列--LSM不同Compaction策略性能分析对比

本文介绍一下参考论文里的Compaction性能分析部分,作者在RocksDB的基础上做了多种策略的改造,然后提出了benchmarking方法论,关注compaction性能的哪些维度,并对结果进行分析。 一、Standardization of Compaction Strategies 1.1 实验平台的选择 作者选择了RocksDB作为…

leetcode 3559. Number of Ways to Assign Edge Weights II

leetcode 3559. Number of Ways to Assign Edge Weights II 1. 解题思路2. 代码实现 题目链接&#xff1a;3559. Number of Ways to Assign Edge Weights II 1. 解题思路 这一题是题目3558. Number of Ways to Assign Edge Weights I的进阶版本。 对于题目3558来说&#xf…

推理模型 vs 非推理模型:核心区别及优劣势解析

推理能力上的差异 推理模型在推理能力方面表现突出,它们擅长通过生成中间步骤和“思维链”逐步解决复杂问题。这意味着面对数学计算、逻辑推理、多跳推断等任务时,推理模型能够将问题分解为若干子步骤,每一步给出推理结果,最终汇总得到答案。这种逐步推导的方式使得推理模…

OPENEULER搭建私有云存储服务器

一、关闭防火墙和selinux 二、下载相关软件 下载nginx&#xff0c;mariadb、php、nextcloud 下载nextcloud&#xff1a; sudo wget https://download.nextcloud.com/server/releases/nextcloud-30.0.1.zip sudo unzip nextcloud-30.0.1.zip -d /var/www/html/ sudo chown -R…

Docker 与微服务架构:从单体应用到容器化微服务的迁移实践

随着软件系统规模和复杂性的日益增长,传统的单体应用(Monolithic Application)在开发效率、部署灵活性和可伸缩性方面逐渐暴露出局限性。微服务架构(Microservice Architecture)作为一种将大型应用拆分为一系列小型、独立、松耦合服务的模式,正成为现代企业构建弹性、敏捷…

【C#】Invalidate()的使用

Invalidate()的使用 Invalidate() 是 C# 中用于通知控件需要重新绘制的方法。它通常用于 Windows Forms 应用程序中&#xff0c;当想要更新控件的显示内容时使用。调用 Invalidate() 方法后&#xff0c;系统会安排对该控件进行重绘&#xff0c;这将导致后续调用 OnPaint 方法&…

我店模式系统开发打造本地生活生态商圈

在当今快节奏的商业环境中&#xff0c;商家们面临着越来越多的挑战&#xff0c;包括市场竞争加剧、消费者需求多样化以及运营效率的提高等。为了应对这些挑战&#xff0c;越来越多的商家开始寻求信息化解决方案&#xff0c;以提升运营效率和客户体验。我的店模式系统平台应运而…

Linux(Ubuntu)新建文件权限继承问题

当你在一个工作目权限为777的文件下&#xff0c;新建一个文件的时候&#xff0c;就有可能发生&#xff0c;新建的这个文件&#xff0c;权限和其他文件&#xff0c;或者工作目录不一致的问题&#xff0c;我们不可能每次新建一个文件&#xff0c;就要 sudo chmod -R 777 /PATH 所…

Vue3和React中插件化设计思想

Vue 3 和 React 都广泛支持插件化设计思想&#xff0c;但因为它们的架构和理念不同&#xff0c;插件化的实现方式也不尽相同。以下分别详细讲解这两者中如何实现插件化&#xff1a; &#x1f7e9; 一、Vue 3 中的插件化实现 Vue 3 继承了 Vue 2 的插件机制&#xff0c;同时增强…

Excel 密码忘记了?巧用PassFab for Excel 解密帮您找回数据!

在工作中&#xff0c;你是否遇到过这样的尴尬时刻&#xff1f;打开重要的 Excel 文件&#xff0c;却发现忘记密码&#xff0c;里面的财务报表、客户数据、项目计划瞬间变成 “加密天书”。重新制作耗时耗力&#xff0c;找专业人员解密又担心数据泄露&#xff0c;这个时候&#…

Vue3 与 Vue2 区别

一、Vue3 与 Vue2 区别 对于生命周期来说&#xff0c;整体上变化不大&#xff0c;只是大部分生命周期钩子名称上 “on”&#xff0c;功能上是类似的。不过有一点需要注意&#xff0c;组合式API的Vue3 中使用生命周期钩子时需要先引入&#xff0c;而 Vue2 在选项API中可以直接…