51单片机的键盘检测原理

2023-05-16

一、独立键盘检测
1、按键的检测原理
单片机的I/O口既可以作为输出也可以作为输入使用,检测按键时用的是输入功能。把按键的一端接地,另一端与单片机的某个I/O口相连,开始时先给该I/O口赋一个高电平,然后让单片机不断地检测该I/O口是否变成了低电平,当按键闭合时,相当于该I/O口通过按键与地相连,变成低电平,程序一旦检测到I/O口变为低电平就说明按键被按下,然后执行相应的指令。
2、实验板原理图
独立按键S2,S3,S4,S5分别连接单片机的P3^4,5,6,7。
在这里插入图片描述
3、去抖动操作
由于使用的是弹性小按键,就是一个机械触点的器件。在按下时会有微观上的机械抖动,反应到电平就是高,低,高,低,抖动的长短与机械特性有关,一般在5~10ms。所以在检测键盘是否按下时要加上去抖动操作。
在这里插入图片描述

4、用数码管的前三位实现000~999的循环计时,按下S2时停止,再次按下开始;按下S3时数值加1,按下S4时数值减1;按下S5时数值清零。

#include <reg52.h>
#define uint unsigned int
#define uchar unsigned char
unsigned char code table[]={0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};
sbit dula=P2^6;
sbit wela=P2^7;
sbit key1=P3^4;
sbit key2=P3^5;
sbit key3=P3^6;
sbit key4=P3^7;
int num=0;
char num1=0;
void delayms(uint xms)
{
	uint i,j;
	for(i=xms;i>0;i--)
		for(j=110;j>0;j--);
}
void keyscan()//键盘扫描函数
{
	if(key1==0)
	{
		delayms(10);
		if(key1==0)
		{
			TR0=~TR0;//启动或关闭定时器
		}
		while(!key1);
	}
	if(key2==0)
	{
		delayms(10);//去抖动操作,如果不加会出现num的值加了好多个,因为循环检测速度非常快
		if(key2==0)
		{
			num++;
			if(num==1000)
				num=0;
		}
		while(!key2);//等待按键释放,因为人为手动按下的过程的时间比单片机检测的时间长很多,
	}//如果不加也会出现num的值加了很多次的现象
	if(key3==0)
	{
		delayms(10);
		if(key3==0)
		{
			num--;
			if(num==-1)
				num=999;
		}
		while(!key3);
	}
	if(key4==0)
	{
		delayms(10);
		if(key4==0)
		{
			num=0;
		}
		while(!key4);
	}
}
void display(uchar ge,uchar shi,uchar bai)
{
	dula=1;
	P0=table[bai];
	dula=0;
	P0=0xff;
	wela=1;
	P0=0xfe;//打开第一个数码管
	wela=0;
	delayms(1);
	wela=1;//关闭数码管,不然快速显示时数码管数字之外的二极管有微弱的亮度
	P0=0xff;
	wela=0;

	dula=1;
	P0=table[shi];
	dula=0;
	P0=0xff;
	wela=1;
	P0=0xfd;
	wela=0;
	delayms(1);
	wela=1;
	P0=0xff;
	wela=0;

	dula=1;
	P0=table[ge];
	dula=0;
	P0=0xff;
	wela=1;
	P0=0xfb;
	wela=0;
	delayms(1);
	wela=1;
	P0=0xff;
	wela=0;
}
void main()
{
	dula=0;
	wela=0;//数码管初始化,开始时关闭所有数码管
	TMOD=0x01;
	TH0=(65536-45872)/256;
	TL0=(65536-45872)%256;
	EA=1;
	ET0=1;
	TR0=1;
	while(1)
	{
		keyscan();//一直在while循环里检测键盘是否被按下
		display(num%10,num/10%10,num/100);
	}
}
void timer0() interrupt 1
{
	TH0=(65536-45872)/256;
	TL0=(65536-45872)%256;
	num1++;
	if(num1==20)//1s
	{
		num1=0;
		num++;
		if(num==1000)
			num=0;
	}
}

总结:独立键盘主要注意两点(1)按下时的去抖动延时delayms函数,大概10ms;(2)松手时的按键释放检测while(!key);等待按键释放。
二、矩阵键盘检测
1、矩阵键盘连接图
结合上面的原理图,实验板上的4乘4的矩阵分别接单片机的P3^0到7口。行线和列线是线与的关系即0&1=0,只要两根线有一根为0则整根线为低电平。独立键盘和单片机连接时每个键盘都占有一个I/O口,当键盘数量较多时单片机的I/O口就不够用了,就引入了矩阵键盘。试验板上是16个按键的4乘4矩阵键盘即4行4列,每行每个按键的一端连接在一起构成行线,每列每个按键的另一端连接在一起构成列线,这样就是4行4列的8根线,就连接到单片机的8个I/O口。
在这里插入图片描述
2、矩阵键盘的检测原理
独立键盘的一端固定为低电平,检测时比较方便。矩阵键盘两端都连接单片机的I/O口,就需要人为的通过单片机送出低电平。检测的时候,先送一行为低电平,其余几行为高电平,这就确定了哪一行,然后立即轮流检测一次各列是否有低电平,如果检测到某一列为低电平就确定了哪一列。用同样的方法轮流送各行一次低电平,再轮流检测一次各列是否变为低电平,这样就可以检测完所有的按键。(也可以将列线置低电平,扫描行线是否有低电平)
3、按下16个矩阵键盘依次在数码管上显示1-16的平方。如按下第一个显示1,第二个显示4,第三个显示9…

#include <reg52.h>
sbit dula=P2^6;
sbit wela=P2^7;
#define uint unsigned int
#define uchar unsigned char
unsigned char code table[]={0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};
void delayms(uint xms)
{
	uint i,j;
	for(i=xms;i>0;i--)
		for(j=110;j>0;j--);
}
void display(int num)//将每个按键要显示的数传递给形参
{
	dula=1;
	P0=table[num/100];//百位
	dula=0;
	P0=0xff;
	wela=1;
	P0=0xfe;
	wela=0;
	delayms(1);
	wela=1;
	P0=0xff;
	wela=0;

	dula=1;
	P0=table[num/10%10];//十位
	dula=0;
	P0=0xff;
	wela=1;
	P0=0xfd;
	wela=0;
	delayms(1);
	wela=1;
	P0=0xff;
	wela=0;
	
	dula=1;
	P0=table[num%10];//个位
	dula=0;
	P0=0xff;
	wela=1;
	P0=0xfb;
	wela=0;
	delayms(1);
	wela=1;
	P0=0xff;
	wela=0;
}
void keyscan()
{
	uchar temp;
	int key;
	P3=0xfe;//将第一行置为0
	temp=P3;
	temp=temp&0xf0;//如果第一行有按键按下,与f0相与之后肯定不是f0,说明被按下
	if(temp!=0xf0)
	{
		delayms(10);//去抖动
		temp=P3;
		temp=temp&0xf0;
		if(temp!=0xf0)
		{
			temp=P3;//将P3口重新赋值给temp,因为之前的temp是相与之后的结果
			switch(temp)
			{
				case 0xee:key=1;break;
				case 0xde:key=2*2;break;
				case 0xbe:key=3*3;break;
				case 0x7e:key=4*4;break;
			}
		}
		while(temp!=0xf0)//检测按键是否释放
		{
			temp=P3;
			temp=temp&0xf0;//比独立按键麻烦一点,就是要一直检测P3口
		}
		display(key);//显示按键要显示的值
	}

	P3=0xfd;//第二行
	temp=P3;
	temp=temp&0xf0;
	if(temp!=0xf0)
	{
		delayms(10);
		temp=P3;
		temp=temp&0xf0;
		if(temp!=0xf0)
		{
			temp=P3;
			switch(temp)
			{
				case 0xed:key=5*5;break;
				case 0xdd:key=6*6;break;
				case 0xbd:key=7*7;break;
				case 0x7d:key=8*8;break;
			}
		}
		while(temp!=0xf0)
		{
			temp=P3;
			temp=temp&0xf0;
		}
		display(key);
	}

	P3=0xfb;//第三行
	temp=P3;
	temp=temp&0xf0;
	if(temp!=0xf0)
	{
		delayms(10);
		temp=P3;
		temp=temp&0xf0;
		if(temp!=0xf0)
		{
			temp=P3;
			switch(temp)
			{
				case 0xeb:key=9*9;break;
				case 0xdb:key=10*10;break;
				case 0xbb:key=11*11;break;
				case 0x7b:key=12*12;break;
			}
		}
		while(temp!=0xf0)
		{
			temp=P3;
			temp=temp&0xf0;
		}
		display(key);
	}

	P3=0xf7;//第四行
	temp=P3;
	temp=temp&0xf0;
	if(temp!=0xf0)
	{
		delayms(10);
		temp=P3;
		temp=temp&0xf0;
		if(temp!=0xf0)
		{
			temp=P3;
			switch(temp)
			{
				case 0xe7:key=13*13;break;
				case 0xd7:key=14*14;break;
				case 0xb7:key=15*15;break;
				case 0x77:key=16*16;break;
			}
		}
		while(temp!=0xf0)
		{
			temp=P3;
			temp=temp&0xf0;
		}
	}
	display(key);
}

void main()
{
	dula=0;
	wela=0;
	while(1)
	{
		keyscan();//在大循环里一直检测按键
	}
}

在按下第四行第四列的按键后显示256
在这里插入图片描述
矩阵键盘关键的代码在于分别将每行置0然后检测每一列,去抖动之后,在检测按键是否释放要一直读取P3口

	P3=0xfe;//将第一行置为0
	temp=P3;
	temp=temp&0xf0;//如果第一行有按键按下,与f0相与之后肯定不是f0,说明被按下
	if(temp!=0xf0)
	{
		delayms(10);//去抖动
		temp=P3;
		temp=temp&0xf0;
		if(temp!=0xf0)
		{
			temp=P3;//将P3口重新赋值给temp,因为之前的temp是相与之后的结果
			switch(temp)
			{
				case 0xee:key=1;break;
				case 0xde:key=2*2;break;
				case 0xbe:key=3*3;break;
				case 0x7e:key=4*4;break;
			}
		}
		while(temp!=0xf0)//检测按键是否释放
		{
			temp=P3;
			temp=temp&0xf0;//比独立按键麻烦一点,就是要一直检测P3口
		}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

51单片机的键盘检测原理 的相关文章

  • FLOPS和FLOPs、GFLOPs区别与计算

    参考FLOPS和FLOPs GFLOPs区别与计算
  • VS远程调试

    文章目录 VS远程调试本地和虚拟机调试准备工作 xff1a 需要注意的几个地方 xff1a VS远程调试 在编程中由于环境 版本等各种原因 xff0c 我们需要模拟出来各种环境来跑不同的版本测试 本地和虚拟机调试 准备工作 xff1a 以V
  • AUTOSAR基础篇之CanNM

    前言 首先 xff0c 问大家几个问题 xff0c 你清楚 xff1a 为什么要引入网络管理呢 xff1f 上电同时启动 xff0c 下电同时关闭 xff0c 它不香吗 xff1f 你知道车上的ECU节点可以分为哪几种类型吗 xff1f 汽
  • CANoe应用案例之DoIP通信

    随着ECU功能和存储容量的不断提高 xff0c 主机厂对于ECU诊断和刷写有了更高的要求 由于带宽的限制 xff0c 传统的汽车总线 xff08 如CAN总线 xff09 存在刷写时间过长的缺点 xff0c 大大降低了生产和维修效率 DoI
  • TRACE32——常用操作

    TRACE 32常用操作 TRACE32软件打开后 xff0c 连上硬件环境 xff0c 我们就可以开始尝试和芯片建立连接 xff0c 并进行基本的调试操作 第一步 xff1a 确认目标板是否上电 第二步 xff1a 打开System Se
  • TRACE32——基于SNOOPer的变量记录

    TRACE32 基于SNOOPer的变量记录 在我们日常调试工作中 xff0c 经常会遇到一种场景 xff1a 对于某些变量或者内存的值 xff0c 希望对其进行监控 当这些变量发生写或者读的时候 xff0c 将这些操作记录下来 xff0c
  • TRACE32——内存填充测试Data.Pattern

    TRACE32 内存填充测试Data Pattern Data Pattern 命令可以用于对内存 xff08 SRAM DDR Flash等 xff09 写入随机值 xff0c 以快速地测试内存是否可以正确读写 命令格式 xff1a 示
  • STM32使用printf重定向

    最近用STM32CubeMX创建了一个demo工程 xff0c 在调试过程中 xff0c printf打印功能一直不能正常打印 xff0c 检查工程中也已将fputc函数进行了实现 奇怪的是用JTAG进行调试时打印恢复了正常 最后发现问题的
  • repo的安装和使用

    前言 Android 采用 Gerrit 提供代码评审服务 xff0c 并且开发了一个客户端工具 repo xff0c 实现多仓库管理 Git 的开发者对服务端的 Git 源码做了扩展 xff0c 使得基于 Git xff08 cgit x
  • repo sync之后不能自动 rebase 的定位

    背景 最近在使用repo sync同步代码时老是报告 xff1a branch xxx is published but not merged and is now 1 commits behind 我之前是由推送过一笔提交 xff0c 但
  • git取指定日期log问题

    库上版本有一个重要bug xff0c 使用了如下命令取一个版本 xff1a repo forall c 39 commitID 61 96 git log before 34 2022 12 09 00 00 00 34 1 pretty
  • ROS读取激光雷达点云数据(RS-Lidar为例)

    一 准备工作 xff1a 1 安装ROS xff08 含有rviz xff09 xff1b 2 安装pcl ros pcl xff08 Point Cloud Library xff09 ros 是ROS中点云和3D几何处理的接口和工具 如
  • 标准预定义的宏

    标准预定义宏是由相关的语言标准规定的 xff0c 所以它们可以在所有执行这些标准的编译器中使用 旧的编译器可能不会提供所有这些宏 它们的名字都以双下划线开头 FILE 这个宏扩展为当前输入文件的名称 xff0c 以 C 字符串常数的形式 这
  • Realsense L515使用

    提示 xff1a 文章写完后 xff0c 目录可以自动生成 xff0c 如何生成可参考右边的帮助文档 文章目录 前言一 主要参考二 基本情况 xff08 一 xff09 ubuntu和ROS情况 xff08 二 xff09 主要步骤 总结
  • 电脑前置摄像头运行ORB-SLAM2 Mono

    ROS本地摄像头 下载安装usb cam包 最好将它放在ROS空间的src文件夹下 xff0c 方便管理 git clone https github com bosch ros pkg usb cam git usb cam cd usb
  • ERROR:cannot launch node of type

    当使用roslaunch启动ros节点时 xff0c 如果出现 ERROR cannot launch node of type 这个错误 xff0c 一般原因是由于没有source bashrc 因此工作空间使用 source bashr
  • Ubuntu 18.04 运行PL-VINS

    代码地址 https span class token operator span span class token comment github com cnqiangfu PL VINS span 安装过程出错参考 PL VINS配置
  • rosbag.bag.ROSBagUnindexedException: Unindexed bag

    rosbag bag ROSBagUnindexedException Unindexed bag ROSBAG
  • d435i 相机和IMU联合标定

    一 前提 我们已经对RGB摄像头和IMU进行了单独标定 参考之前博文 RGB标定 IMU kalibr官方WIKI Kalibr 二 准备标定文件 2 1 标定板yaml文件 标定下载链接 Aprilgrid 6x6 0 5x0 5 m u
  • opencv版权nofree问题

    找到3 1 0中cpp文件拷贝到3 4 12中 xff0c 重新cmake编译 xff0c 即可使用 nofree解决方案

随机推荐