一、矩阵按键
矩阵按键就是独立按键的结合体,矩阵按键的优点是提高了I/O端口的利用率,用M+N个I/O端口就可以控制M×N个独立按键。而一个独立按键就需要一个I/O端口来控制,I/O端口的利用率很低。
如上图所示,每列按键引出一条线由一个I/O端口控制,每行按键引出一条线由一个I/O端口控制。
二、按键扫描原理
行列扫描法:以上图矩阵按键电路图,端口高四位是行,低四位是列。
- 确认行,首先给端口高四位为高电平,低四位为低电平,即11110000,然后检测高四位接收到的值是否全为1。若不全为1,则证明有键被按下。以按键S1为例,当按键S1被按下时,P17端口与P13端口通路,此时P17端口为低电平。如果检测到P1端口数据为01110000,则第一行有按键被按下,其他行同理。
- 确认列,与确认行原理相似,给P1端口低四位为高电平,高四位为低电平,即00001111。检测低四位是否全为1,若低四位为0111,则第一列有按键被按下。根据确认行和确认列就可以判断是哪个坐标的按键被按下。
三、实验验证
基于STC89C52芯片的普中科技HC6800-EM3 V2.2开发板。
数码管原理图
Proteus仿真图
烧写开发板时程序出错,临时做了个仿真图。
程序
#include <reg51.h>
unsigned char code table[] = {0xc0, 0xf9, 0xa4, 0xb0, 0x99, 0x92, 0x82, 0xf8, 0x80, 0x90 ,0x88,0x03,0x46,0xa1,0x86,0x8e }; // 共阳极数码管0~F
unsigned char keycode; // keycode用来临时存放P1端口的值
unsigned int KeyValue,H_value,L_value; // KeyValue为计算出的按键号, H_value为第几行, L_value为第几列
void delay10ms(void)
{
unsigned char a,b,c;
for(c=1;c>0;c--)
for(b=38;b>0;b--)
for(a=130;a>0;a--);
}
void ScanKeyboard()
{
H_value =0;
L_value =0;
P1 = 0xf0; // 高四位1111 低四位0000 扫描行
if(P1 != 0xf0)
{
delay10ms(); // 消抖
if(P1 != 0xf0)
{
keycode = P1;
switch(keycode)
{
case(0x70): H_value = 1; break;
case(0xb0): H_value = 2; break;
case(0xd0): H_value = 3; break;
case(0xe0): H_value = 4; break;
default: break;
}
}
}
P1 = 0x0f; // 高四位0000 低四位1111 扫描列
keycode = P1;
if(P1 != 0x0f)
{
delay10ms();
if(P1 != 0x0f)
{
keycode = P1;
switch(keycode)
{
case(0x07): L_value = 1; break;
case(0x0b): L_value = 2; break;
case(0x0d): L_value = 3; break;
case(0x0e): L_value = 4; break;
default: break;
}
}
}
KeyValue = (H_value-1)*4 + L_value-1; // 通过第几行第几列计算出按键的坐标,也能得到按键号
}
void main()
{
P3 = 0xff;
while(1)
{
ScanKeyboard();
if(H_value!= 0 && L_value!= 0)
{
P3 = table[KeyValue]; // 控制数码管显示
}
}
}