单片机按键扫描程序,可以单击、双击、长按,使用状态机,无延时,不阻塞。
- 根据按下时的时长、间隔来判断是否是连按或者长按。
- 当连按间隔很短时,计录连按次数
- 超过连接间隔时,回报按下次数
- 根据按键次数自行判断是单击、双击、三击、四击。。。最多记录15击。
结构体版:
#define KEY_CHANNEL_COUNT (6 + 8 + 8)
struct keyInfo
{uint8_t act : 4; // 按了多少次,最多连按15次uint8_t down : 1; // 按下了uint8_t up : 1; // 松开了uint8_t longPress : 1; // 长按了uint8_t io : 1; // 按键IO状态uint8_t intervalTime; // 连按间隔时间uint8_t holdTime; // 长按时间uint8_t duration; // 按键次数保持时间,超过后,act清零
};
struct keyInfo keyValues[KEY_CHANNEL_COUNT] = {0};void button_trace_handle(void *p)
{static uint32_t lastTime = 0;const uint8_t KEYSCAN_INTERVAL_TIME = 10; // 按键扫描间隔时间const uint8_t LONG_PRESS_TIME = 100; // 长按多久生效, 实际时间为,下面同理 LONG_PRESS_TIME * KEYSCAN_INTERVAL_TIMEconst uint8_t INTERVAL_TIME_SET = 20; // 两次按键检测超时const uint8_t ANTI_SHAKE_TIME = 2; // 按键防抖检测超时/*** @brief 需要实现millis()函数,系统毫秒计时器。* */if (millis()> lastTime + KEYSCAN_INTERVAL_TIME){lastTime = millis(); }else{return;}/**给按键IO赋值, 有多个按键就传多少个, 自己实现ic_read函数 */for (uint8_t i = 0; i < KEY_CHANNEL_COUNT; i++){keyValues[i].io = io_read(i);}for (uint8_t i = 0; i < KEY_CHANNEL_COUNT; i++){if (keyValues[i].io)//按下了{if (keyValues[i].holdTime < LONG_PRESS_TIME){keyValues[i].holdTime++;if (keyValues[i].holdTime >= ANTI_SHAKE_TIME)//防抖{keyValues[i].down = 1;keyValues[i].intervalTime = INTERVAL_TIME_SET;}}else //长按了,会一直标记,直到松开{keyValues[i].longPress = 1;}}else{//松开了keyValues[i].holdTime = 0;keyValues[i].longPress = 0;if (keyValues[i].down)//按下过了{if (!keyValues[i].longPress)//不是长按keyValues[i].act++;//按下次数+1keyValues[i].down = 0;}if (keyValues[i].intervalTime)//连按超时{keyValues[i].intervalTime--;if (keyValues[i].intervalTime == 1){LOG_D("key[%d] act:%d", i, keyValues[i].act);//打印哪个按键按了多少次keyValues[i].duration = 10;}}}if (keyValues[i].duration >= 1){keyValues[i].duration--;if (keyValues[i].duration == 1)//按键次数保持时间到{keyValues[i].act = 0;}}}
}
无结构体版,更方便移到51单片机上
#define KEY_DOWN_MASK 0X80/**按下标记*/
#define KEY_LONG_PRESS_MASK 0X40/**长按标记 */
#define KEY_TIMEOUT_MASK 0X10/**超时标志,此时返回按键值*/
#define KEY_TIEMES_MASK 0X0F/**按了多少次 */
#define KEY_VALUE(x) (0x0001<<(x))#define KEY_COUNTS 5void keyScanPro()
{const uint8_t SHORT_PRESS_TIME = 25;const uint16_t LONG_PRESS_TIME = 150;const uint8_t IS_KEY_DOWN = 0X80;/**按下了 */const uint8_t IS_LONG_PRESS = 0X40;/**长按了 */const uint8_t IS_TIME_OUT = 0X10;/**退好久没按 */static uint8_t keyActionHold = 0;static uint8_t pressTimesRecord[KEY_COUNTS] = { 0 };static uint8_t pressTime[KEY_COUNTS] = { 0 };static uint16_t longPressTime[KEY_COUNTS] = { 0 };const uint16_t channel_keyScan_map[KEY_COUNTS] = { DEF_SET_BIT0,DEF_SET_BIT1,DEF_SET_BIT2,DEF_SET_BIT3,DEF_SET_BIT4 };/**A,B,C,D,E,F,G,H,I,J,K,L对就的键值*/uint8_t i;keyValue = KP; keyValue <<= 1;keyValue |= !K1; keyValue <<= 1;keyValue |= !K2; keyValue <<= 1;keyValue |= !K3; keyValue <<= 1;keyValue |= !K4;if (keyValue != keyValuePre){ResetSystemShutdownCountdown();keyValuePre = keyValue;}// LOG("keyValue:%d\n",(int)keyValue);if (keyAction){if (keyActionHold++ > 100){keyActionHold = 0;keyAction = 0;}}for (i = 0; i < KEY_COUNTS; i++){if (keyValue & channel_keyScan_map[i]){//按下了if (longPressTime[i] < LONG_PRESS_TIME){longPressTime[i]++;pressTime[i] = SHORT_PRESS_TIME;pressTimesRecord[i] |= IS_KEY_DOWN;}else{pressTimesRecord[i] |= IS_LONG_PRESS;pressTimesRecord[i] |= IS_TIME_OUT;pressTimesRecord[i] &= ~IS_KEY_DOWN; //取消标记高位keyLongPress |= 1 << i;keyAction |= ((i + 1) << 8);if (keyValue & PWR_KEY_VALUE){LOG("System shutting down ...");SYS_PWR_SHUTDOWN();while (1);}LOG("long press:%d\n", (int)keyLongPress);// if (pwrKeyLongPressCb) pwrKeyLongPressCb();// else pwrKeyLongPressCbDefault();}}else{//松开了longPressTime[i] = 0;keyLongPress &= ~(1 << i);if (pressTimesRecord[i] & IS_KEY_DOWN){//高位标记过,即按下过keyActionHold = 0;pressTimesRecord[i] &= ~IS_KEY_DOWN; //取消标记高位if ((pressTimesRecord[i] & KEY_TIEMES_MASK) < 15){uint8_t ptc = 0;pressTimesRecord[i]++;ptc = pressTimesRecord[i] & KEY_TIEMES_MASK;ptc = ptc > 7 ? 7 : ptc;// speaker_out(music_note_freq[ptc], 100);}}if (pressTime[i] > 0){if (pressTime[i] == 1){pressTimesRecord[i] |= IS_TIME_OUT;//BIT4 为检测时间到}pressTime[i]--;}if (pressTimesRecord[i] & IS_TIME_OUT){if (pressTimesRecord[i] & IS_LONG_PRESS){// rt_kprintf("Long press:%d \n", i);pressTimesRecord[i] &= ~IS_LONG_PRESS;}else if (pressTimesRecord[i] & KEY_TIEMES_MASK){uint8_t ptc = pressTimesRecord[i] & KEY_TIEMES_MASK;keyAction |= (i + 1) << 4 | ptc;LOG("keyAction:%x\n", (int)keyAction);// struct pwrKeyActList* p;// p = &pwrKeyActListHead;// do {// // LOG_D("P:0x%08X", p);// if (p->cb)// {// p->cb(ptc);// }// p = p->next;// } while (p);}// LOG("Press:%d - %d\n", (int)i, (int)(pressTimesRecord[i] & KEY_TIEMES_MASK));;pressTimesRecord[i] = 0;}}}
}
相关文章:
单片机按键扫描程序,可以单击、双击、长按,使用状态机,无延时,不阻塞。
根据按下时的时长、间隔来判断是否是连按或者长按。当连按间隔很短时,计录连按次数超过连接间隔时,回报按下次数根据按键次数自行判断是单击、双击、三击、四击。。。最多记录15击。 结构体版: #define KEY_CHANNEL_COUNT (6 8 8) struct…...
Django中自定义模板字符串
首先在目录中创建名为 templatetags 的文件夹 然后在文件中创建 python 文件 然后再相应的 HTML 模板文件中导入 {% load python的文件名 %} 然后在 HTML 文件中通过进行使用 {% 函数名 %} python文件中的代码 首先导入 from django import templateregister template…...
暴雨总裁孙辉:混合式人工智能是大势所趋
当前,生成式人工智能技术的突破和发展,大大加速了各行各业的数字化、智能化进程。在充满不确定性的全球发展环境下,人工智能已成为驱动未来增长最确定、最核心的力量。 对此,暴雨总裁孙辉认为,2025年,人工…...
【小制作】米家模拟手指点击
代码功能解释 这段代码是一个基于Arduino平台的控制程序,主要功能包括: 初始化:设置引脚模式、初始化编码器、舵机和EEPROM。按键检测:处理按钮的单击、双击和长按事件,并根据事件执行相应操作。编码器更新ÿ…...
【深度学习入门_基础篇】线性代数本质
开坑本部分主要为基础知识复习,新开坑中,学习记录自用。 学习目标: 熟悉向量、线性组合、线性变换、基变换、矩阵运算、逆函数、秩、列空间、零空间、范式、特征指、特征向量等含义与应用。 强烈推荐此视频: 【官方双语/合集】…...
047_小驰私房菜_Qcom 8系列,Jpeg GPU 旋转
【问题背景】 横屏模式下,发现有些三方app拍照旋转了90度。 【修改策略】 adb shell setprop endor.debug.camera.overrideGPURotationUsecase 1 或者在/vendor/etc/camera/camxoverridesettings.txt 里面添加如下内容 overrideGPURotationUsecase1 【解释】 Ga…...
Elasticsearch 操作文档对数据的增删改查操作 索引库文档 操作数据 CRUD
介绍 在 Elasticsearch 中,文档的增、删、改、查操作是核心的基本功能。Elasticsearch 使用 RESTful API 提供这些操作,通常通过 HTTP 请求与 Elasticsearch 集群进行交互。 索引库 {"mappings": {"properties": {"title&qu…...
最新MySQL面试题(2025超详细版)
2025最新超详细MySQL面试题 文章目录 2025最新超详细MySQL面试题[toc]一、 SQL 和基本操作1. SQL的执行顺序2. 如何优化MySQL查询3. 常用的聚合函数4. 数据库事务5. 事务的四大特性(ACID)6. 视图7. MySQL中使用LIMIT子句进行分页8. MySQL中使用变量和用户定义的函数9. MySQL中的…...
使用MPTCP+BBR进行数据传输,让网络又快又稳
1.前言 在前文《链路聚合技术——多路径传输Multipath TCP(MPTCP)快速实践》中我们使用mptcpize run命令实现了两个节点间通信使用MPTCP协议进行传输,并实现了传输速率的聚合。 实际应用中更推荐原生支持mptcp的应用,在MPTCP官网中可以看到如TCPDump、…...
滴滴数据分析80道面试题及参考答案
如何衡量分类好坏? 衡量分类好坏有多种方法,常用的有准确率、精确率、召回率、F1 值、ROC 曲线与 AUC 值等。 准确率:是指分类正确的样本数占总样本数的比例,计算公式为:准确率 = (分类正确的样本数)/(总样本数)。准确率越高,说明分类器整体的分类效果越好,但在正负…...
基于物联网疫苗冷链物流监测系统设计
1. 项目开发背景 随着全球对疫苗运输要求的提高,特别是针对温度敏感型药品(如疫苗)的冷链管理,如何保证疫苗在运输过程中的温度、湿度、震动等环境因素的稳定性已成为亟需解决的问题。疫苗运输过程中,任何温度或湿度的…...
计算机网络基础(7)中科大郑铨老师笔记
应用层 目标: 网络应用的 原理:网络应用协议的概念和实现方面 传输层的服务模型 客户-服务器模式 对等模式(peerto-peer) 内容分发网络 网络应用的 实例:互联网流行的应用层协 议 HTTP FTP SMTP / POP3 / IMAP DNS…...
GOGOGO 抽象
抽象其实也算面向对象特征之一 抽象 含义:当多个子类中的共性向上提取,父类中不知道如何写具体实现,因为提取的共性并不一定能解决子类中实现的功能【同结构不一定同实现代码体】,就需要抽象概念 作用 父类只抽取结构ÿ…...
STM32-笔记26-WWDG窗口看门狗
一、简介 窗口看门狗用于监测单片机程序运行时效是否精准,主要检测软件异常,一般用于需要精准检测程序运行时间的场合。 窗口看门狗的本质是一个能产生系统复位信号和提前唤醒中断的6位计数器(有的地方说7位。其实都无所谓࿰…...
10.装饰器
装饰器的基本用法创建简单的装饰器带参数的装饰器装饰器链类装饰器内置装饰器 定义:装饰器是一个函数,它接受另一个函数作为参数,并返回一个新的函数。装饰器通常用于在函数执行前后添加额外的功能,如日志记录、权限检查、性能测试等。 1.基…...
uniapp H5页面实现懒加载
在 uniapp 中,要在小的 view 内实现列表懒加载,可以通过以下步骤来实现: 使用 scroll-view 组件来创建一个可滚动的区域。在 scroll-view内 部放置一个list组件,用于显示数据列表。监听 scroll-view 的滚动事件,当滚动…...
STM32使用UART发送字符串与printf输出重定向
首先我们先看STM32F103C8T6的电路图 由图可知,其PA9和PA10引脚分别为UART的TX和RX(注意:这个电路图是错误的,应该是PA9是X而PA9是RX,我们看下图的官方文件可以看出),那么接下来我们应该找到该引脚的定义是什么…...
NLP初识
目录 0简介一、自然语言概述1. 什么是NLP?2. NLP常用工具0简介 NLP系列开始更新了!!!这个系列主要会介绍一些NLP的基础概念,比如RNN、LSTM、GRU等内容,重头戏放在大语言模型的基础讲解上,其中大语言模型的分享主要由两个方面:1.基础结构(Seq2Seq,Attention,Transfor…...
解決當前IP地址僅適用於本地網路
想要解決“當前IP地址僅適用於本地網路”其實並不困難。本篇文章將介紹其發生的原因以及如何解決。 “僅限本地網路”是什麼意思? 當IP地址為“僅限本地網路”時,意味著設備正在使用私人網路內部IP地址,但無法連接到互聯網。如果將本地IP視…...
Eplan 项目结构(高层代号、安装地点、位置代号)
Eplan中的项目结构分为3个层次: (1)功能面结构。指明这个系统的功能,有什么用途。在EPlan中,指的就是"高层代号()"。 一般指的是线体。 (2)位置面结构。指明该…...
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…...
【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型
摘要 拍照搜题系统采用“三层管道(多模态 OCR → 语义检索 → 答案渲染)、两级检索(倒排 BM25 向量 HNSW)并以大语言模型兜底”的整体框架: 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后,分别用…...
通过Wrangler CLI在worker中创建数据库和表
官方使用文档:Getting started Cloudflare D1 docs 创建数据库 在命令行中执行完成之后,会在本地和远程创建数据库: npx wranglerlatest d1 create prod-d1-tutorial 在cf中就可以看到数据库: 现在,您的Cloudfla…...
Vue3 + Element Plus + TypeScript中el-transfer穿梭框组件使用详解及示例
使用详解 Element Plus 的 el-transfer 组件是一个强大的穿梭框组件,常用于在两个集合之间进行数据转移,如权限分配、数据选择等场景。下面我将详细介绍其用法并提供一个完整示例。 核心特性与用法 基本属性 v-model:绑定右侧列表的值&…...
Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务
通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...
抖音增长新引擎:品融电商,一站式全案代运营领跑者
抖音增长新引擎:品融电商,一站式全案代运营领跑者 在抖音这个日活超7亿的流量汪洋中,品牌如何破浪前行?自建团队成本高、效果难控;碎片化运营又难成合力——这正是许多企业面临的增长困局。品融电商以「抖音全案代运营…...
React19源码系列之 事件插件系统
事件类别 事件类型 定义 文档 Event Event 接口表示在 EventTarget 上出现的事件。 Event - Web API | MDN UIEvent UIEvent 接口表示简单的用户界面事件。 UIEvent - Web API | MDN KeyboardEvent KeyboardEvent 对象描述了用户与键盘的交互。 KeyboardEvent - Web…...
HTML前端开发:JavaScript 常用事件详解
作为前端开发的核心,JavaScript 事件是用户与网页交互的基础。以下是常见事件的详细说明和用法示例: 1. onclick - 点击事件 当元素被单击时触发(左键点击) button.onclick function() {alert("按钮被点击了!&…...
CMake控制VS2022项目文件分组
我们可以通过 CMake 控制源文件的组织结构,使它们在 VS 解决方案资源管理器中以“组”(Filter)的形式进行分类展示。 🎯 目标 通过 CMake 脚本将 .cpp、.h 等源文件分组显示在 Visual Studio 2022 的解决方案资源管理器中。 ✅ 支持的方法汇总(共4种) 方法描述是否推荐…...
分布式增量爬虫实现方案
之前我们在讨论的是分布式爬虫如何实现增量爬取。增量爬虫的目标是只爬取新产生或发生变化的页面,避免重复抓取,以节省资源和时间。 在分布式环境下,增量爬虫的实现需要考虑多个爬虫节点之间的协调和去重。 另一种思路:将增量判…...
