单片机——矩阵按键模块
主要目的
学会按键扫描
1.延时函数
延时函数部分详见链接: 单片机控制一盏灯的亮与灭程序解释
void delay (uint k) //定义延时函数{uint i,j;for(i<0;i<k;i++){for(j=0;j<113;j++){;}}}
这个程序里面的延时函数的目的是按键消抖。
2.按键扫描模块
这是本次实验的重点,将详细介绍。
先来观察矩阵按键模块的连接

总共8个口。
先实现关于列的扫描,设置初始值(0xf0),从高到低为
| 端口 | Value |
|---|---|
| P3.7 | 1 |
| P3.6 | 1 |
| P3.5 | 1 |
| P3.4 | 1 |
| P3.3 | 0 |
| P3.2 | 0 |
| P3.1 | 0 |
| P3.0 | 0 |
那么当按下按键0,4,8,c时,P3.4变成了0,于是P3就变成了0xe0
| 端口 | Value |
|---|---|
| P3.7 | 1 |
| P3.6 | 1 |
| P3.5 | 1 |
| P3.4 | 0 |
| P3.3 | 0 |
| P3.2 | 0 |
| P3.1 | 0 |
| P3.0 | 0 |
从而由0xe0我们可以判断是那一列按键按下,但还是不知道具体是哪一个按键按下,于是我们需要继续进行行扫描。
此时初始化P3
| 端口 | Value |
|---|---|
| P3.7 | 0 |
| P3.6 | 0 |
| P3.5 | 0 |
| P3.4 | 0 |
| P3.3 | 1 |
| P3.2 | 1 |
| P3.1 | 1 |
| P3.0 | 1 |

同时现在当按键0按下后
| 端口 | Value |
|---|---|
| P3.7 | 0 |
| P3.6 | 0 |
| P3.5 | 0 |
| P3.4 | 0 |
| P3.3 | 1 |
| P3.2 | 1 |
| P3.1 | 1 |
| P3.0 | 0 |
经过行列扫描之后就可以确定具体是哪个按键按下,有了上面的基础之后,我们来学习按键扫描的程序。
void keyscan() //这个实验的重点{ uchar a; P3=0xf0;if(P3!=0xf0){delay(10);if(P3!=0xf0){P3=0xf0; //进行列扫描switch(P3){case(0xe0):keynumber=0;break; //11100000case(0xd0):keynumber=1;break;//11010000case(0xb0):keynumber=2;break;//10110000case(0x70):keynumber=3;break;//01110000}P3=0x0f;switch(P3){case(0x0e):keynumber=keynumber;break; //switch case后面是冒号case(0x0d):keynumber=keynumber+4;break;case(0x0b):keynumber=keynumber+8;break;case(0x07):keynumber=keynumber+12;break;}while ((a<50)&&(P3!=0x0f)){delay(10);a++ ;} }}}
程序的逻辑是
让P3的高四位为0,进行列检测,为什么是列检测呢,很简单,因为区分不出行。
如果按键按下,延时10ms,判断是不是按键抖动,延时之后继续判断,发现仍然不为0xf0,说明按键确实按下了,然后还是给P3幅初始值0xf0,接下来进行switch,case,不同P3情况下,对设置的keynumber进行赋值,记得每一个case结束要加break。
只有列扫描,无法判断是哪个按键按下,接下来进行行扫描。
基本原理和上面一样,但是注意按键值的变化。
最后当两个情况同时不成立后跳出循环。这两个情况就是a>50和P3=0xf0,具体解释就是a>50,按键还没有松开,就认为松开了,P3=0xf0按键按下之后松开了。
3.数码管显示模块
这一部分也相对简单,我的博客里面也有相应的博文。在这里主要就是P0口接的是数码管。这里进行段选和位选。段选是根据keynumber的值,位选这里选了六个数码管,则对应11000000,再转换为16进制。

void display(uchar num){P0=table[num];duan=1;duan=0;P0=0xc0;//11000000wei=1;wei=0;}
4.主函数
主要是设置段和位初始状态,之后调用前面的按键扫描和数码管显示模块。
void main(){duan=0;wei=0;while(1){keyscan();display(keynumber);}}
完整代码
#include <reg52.h>
#define uchar unsigned char
#define uint unsigned int //注意宏定义不能加分号sbit duan = P2^6;
sbit wei = P2^7;uchar keynumber;
uchar code table[]= {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};void delay (uint k) //定义延时函数{uint i,j;for(i<0;i<k;i++){for(j=0;j<113;j++){;}}}void keyscan() //这个实验的重点{ uchar a; P3=0xf0;if(P3!=0xf0){delay(10);if(P3!=0xf0){P3=0xf0; //进行列扫描switch(P3){case(0xe0):keynumber=0;break; //11100000case(0xd0):keynumber=1;break;//11010000case(0xb0):keynumber=2;break;//10110000case(0x70):keynumber=3;break;//01110000}P3=0x0f;switch(P3){case(0x0e):keynumber=keynumber;break; //switch case后面是冒号case(0x0d):keynumber=keynumber+4;break;case(0x0b):keynumber=keynumber+8;break;case(0x07):keynumber=keynumber+12;break;}while ((a<50)&&(P3!=0x0f)){delay(10);a++ ;} }}}void display(uchar num){P0=table[num];duan=1;duan=0;P0=0xc0;//11000000wei=1;wei=0;}void main(){duan=0;wei=0;while(1){keyscan();display(keynumber);}}
运行结果
无法一一展示,这里只展示一部分。

好啦,今天矩阵按键模块的学习就到这里啦。你学废了吗?有问题的话,欢迎共同交流。最近在准备研究生复试,内容比较粗糙,但个人比较喜欢有输出的学习。以后有机会的话,会继续更新完善的!!!
相关文章:
单片机——矩阵按键模块
主要目的 学会按键扫描 1.延时函数 延时函数部分详见链接: 单片机控制一盏灯的亮与灭程序解释 void delay (uint k) //定义延时函数{uint i,j;for(i<0;i<k;i){for(j0;j<113;j){;}}}这个程序里面的延时函数的目的是按键消抖。 2.按键扫描模块 这是本次实验的重点&a…...
Android学习之网络操作
网络操作 Android平台下的原生网络操作可以分为以下几步: 创建URL对象;通过URL对象获取HttpURLConnection对象;通过HttpURLConnection对象设置请求头键值对、网络连接超时时间等;通过HttpURLConnection对象的connect()方法建立网…...
Delphi XE开发android开发环境搭建
Delphi xe为使用Delphi作为开发工具的程序员,提供了开发APP的便捷工具,它的开发环境也是非常容易搭建,这里我简述一下Android的开发环境搭建,Delphi XE开发Android程序的开发环境需要三个软件支持:Java SE Development开发环境、Android SDK和Android Ndk开发环境。 1、安…...
flink入门-流处理
入门需要掌握:从入门demo理解、flink 系统架构(看几个关键组件)、安装、使用flink的命令跑jar包flink的webUI 界面的监控、常见错误、调优 一、入门demo:统计单词个数 0、单词txt 文本内容(words.txt): hello world …...
【数据结构】单链表中,如何实现 将链表中所有结点的链接方向“原地”逆转
一.实现一个单链表(无头单向不循环) 我们首先实现一个无头单向不循环单链表。 写出基本的增删查改功能,以及其它的一些功能(可忽略)。 #include<stdio.h> #include<assert.h> #include<stdlib.h>…...
摘花生(简单DP)
Hello Kitty想摘点花生送给她喜欢的米老鼠。她来到一片有网格状道路的矩形花生地(如下图),从西北角进去,东南角出来。地里每个道路的交叉点上都有种着一株花生苗,上面有若干颗花生,经过一株花生苗就能摘走该它上面所有的花生。Hel…...
2022济南大学acm新生赛题解
通过答题情况的难度系数: 签到:A 简单:BL 中等:D 困难:CM 极难:KNO A-和 算出n个数的和判断正负性即可!!! 发现很多同学的代码错误:要么sum未赋初值&…...
策略模式教程
策略模式是一种行为型设计模式,它允许在运行时根据不同的情况选择不同的算法实现,从而使得算法可以独立于客户端而变化。本文将介绍策略模式的概念、应用场景、优点和缺点,并提供最佳的代码实践。本文的代码实现将使用Java语言,但…...
什么是刺猬理念
一、什么是刺猬理念刺猬理念是指把复杂的世界简化成单个有组织性的观点,一条基本原则或一个基本理念,发挥统帅和指导作用。核心是把事情简单化,把所有的挑战和进退维谷的局面压缩为简单的。二、刺猬理念的寓言故事狐狸是一种狡猾的动物&#…...
RPC通信相关
RPCRPC, 远程过程调用(Remote Procedure Call,RPC)是一个计算机通信协议,该协议允许运行于一台计算机的程序程调用另一台计算机的上的程序。通俗讲,RPC通过把网络通讯抽象为远程的过程调用,调用远程的过程就…...
Node.js + MongoDB 搭建博客 -- 登录页面
准备工作 安装Node.js安装express等相关库MongoDB数据库电脑系统:win11 功能分析 搭建一个简单的具有多人注册、登录、发表文章以及登出功能的博客。 设计目标 未登录:主页左侧导航栏显示home、login、register,右侧显示已发表的文章、发…...
互联网新理念,对于WEB 3.0 你怎么看?
WEB 3.0 这个名词走进大众视野已经有一段时间了,也曾在各个圈子里火热一时,至今各大互联网企业任旧在 WEB 3.0 上不断探索。但关于 WEB 3.0 是什么这个问题,其实大部分人都没有一个比较明确的认知,包括区块链和元宇宙等相关行业的…...
Git使用教程:最详细、最傻瓜、最浅显、真正手把手教
GITGIT版本控制版本控制的意义分布式图形化客户端环境搭建仓库的操作分支使用场景命令远程仓库操作生成公钥命令冲突忽略列表的配置时机配置方式版本回退练习:GIT 版本控制 把文件系统中的文件,按照修改的版本进行记录,进行管理的操作。 版…...
【面试题】Redis面试题汇总(无解答)
Redis为何这么快?缓存问题及解决入库和缓存策略问题及处理redis数据类型缓存过期删除策略内存淘汰机制Redis 回收进程如何工作的?Redis持久化RDB和AOFredis流式pipeline处理原生批命令 (mset, mget) 与 Pipeline 区别?Pipeline 有什么好处,为…...
RHCSA-用户和组管理和文件系统权限(3.11)
目录 用户(UID) 用户类别(UID): 用户的增删改查: 修改用户密码: 查看用户是否存在: 组(GID) 组的增删改查: 设置组密码: 用户…...
RK3588平台开发系列讲解(同步与互斥篇)信号量介绍
平台内核版本安卓版本RK3588Linux 5.10Android 12文章目录 一、信号量介绍二、信号量API1、结构体2、API三、函数调用流程沉淀、分享、成长,让自己和他人都能有所收获!😄 📢上一章我们看了自旋锁的原理,本章我们一起学习下信号量的用法。 一、信号量介绍 和自旋锁一样,…...
One-YOLOv5 v1.2.0发布:支持分类、检测、实例分割
One-YOLOv5 v1.2.0正式发布。完整更新列表请查看链接:https://github.com/Oneflow-Inc/one-yolov5/releases/tag/v1.2.0,欢迎体验新版本,期待你的反馈。 1 新版本特性 1. 同步了Ultralytics YOLOv5的上游分支v7.0,同时支持分类、目…...
Zookeeper的Java API操作
Zookeeper的Java API操作一、先启动Zookeeper集群二、IDEA 环境搭建三、创建子节点四、获取子节点并监听节点变化五、判断 Znode 是否存在六、Watcher工作流程一、先启动Zookeeper集群 二、IDEA 环境搭建 1.创建一个Maven工程:ZookeeperProject 2.在pom.xml文件添…...
Web3:前端知识和后端知识基础
三.Web3:前端知识和后端知识基础 1.了解前端开发 2.了解JSP 3.了解JAVAWeb的三大组件 4.Servlet的使用 5.Filter的使用 6.了解thymeleaf 未更新 三.Web3:前端知识和后端知识基础 1.了解前端开发 ①前端架构 HTML超文本标记语言CSS层叠样式表JavaS...
调试射频TX和rx实验工程出现的问题与反思
1.今天用ADS仿真 发现 加上SMA 插损就到了4db,但是直接用传输线就在1db以内 这个问题我目前想到的排查思路是换成IPEX, 换成IPEX插损就变成2db 拿最新的7626去看 看到上面是SMA-3G 小针 还是结合参考的demo PCB来看 2.用射频的ipex测试LNA 发现校准…...
Linux链表操作全解析
Linux C语言链表深度解析与实战技巧 一、链表基础概念与内核链表优势1.1 为什么使用链表?1.2 Linux 内核链表与用户态链表的区别 二、内核链表结构与宏解析常用宏/函数 三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势5.1 插入效率5.2 安全…...
论文解读:交大港大上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架(二)
HoST框架核心实现方法详解 - 论文深度解读(第二部分) 《Learning Humanoid Standing-up Control across Diverse Postures》 系列文章: 论文深度解读 + 算法与代码分析(二) 作者机构: 上海AI Lab, 上海交通大学, 香港大学, 浙江大学, 香港中文大学 论文主题: 人形机器人…...
【Oracle APEX开发小技巧12】
有如下需求: 有一个问题反馈页面,要实现在apex页面展示能直观看到反馈时间超过7天未处理的数据,方便管理员及时处理反馈。 我的方法:直接将逻辑写在SQL中,这样可以直接在页面展示 完整代码: SELECTSF.FE…...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...
MySQL 隔离级别:脏读、幻读及不可重复读的原理与示例
一、MySQL 隔离级别 MySQL 提供了四种隔离级别,用于控制事务之间的并发访问以及数据的可见性,不同隔离级别对脏读、幻读、不可重复读这几种并发数据问题有着不同的处理方式,具体如下: 隔离级别脏读不可重复读幻读性能特点及锁机制读未提交(READ UNCOMMITTED)允许出现允许…...
.Net Framework 4/C# 关键字(非常用,持续更新...)
一、is 关键字 is 关键字用于检查对象是否于给定类型兼容,如果兼容将返回 true,如果不兼容则返回 false,在进行类型转换前,可以先使用 is 关键字判断对象是否与指定类型兼容,如果兼容才进行转换,这样的转换是安全的。 例如有:首先创建一个字符串对象,然后将字符串对象隐…...
HashMap中的put方法执行流程(流程图)
1 put操作整体流程 HashMap 的 put 操作是其最核心的功能之一。在 JDK 1.8 及以后版本中,其主要逻辑封装在 putVal 这个内部方法中。整个过程大致如下: 初始判断与哈希计算: 首先,putVal 方法会检查当前的 table(也就…...
基于Java+MySQL实现(GUI)客户管理系统
客户资料管理系统的设计与实现 第一章 需求分析 1.1 需求总体介绍 本项目为了方便维护客户信息为了方便维护客户信息,对客户进行统一管理,可以把所有客户信息录入系统,进行维护和统计功能。可通过文件的方式保存相关录入数据,对…...
Unity UGUI Button事件流程
场景结构 测试代码 public class TestBtn : MonoBehaviour {void Start(){var btn GetComponent<Button>();btn.onClick.AddListener(OnClick);}private void OnClick(){Debug.Log("666");}}当添加事件时 // 实例化一个ButtonClickedEvent的事件 [Formerl…...
Linux-进程间的通信
1、IPC: Inter Process Communication(进程间通信): 由于每个进程在操作系统中有独立的地址空间,它们不能像线程那样直接访问彼此的内存,所以必须通过某种方式进行通信。 常见的 IPC 方式包括&#…...
