0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

矩阵键盘的按键识别方法_矩阵键盘扫描程序

姚小熊27 来源:网络整理 作者:网络整理 2020-04-20 09:39 次阅读

矩阵键盘的按键识别方法

矩阵键盘的按键识别方法来自简单日记网精选推荐。在学习有关矩阵键盘的时候,往往要学会矩阵键盘的按键识别方法,那么矩阵键盘的按键识别方法有哪些呢?小编带着你来了解。

方法一:行扫描法

1、判断键盘中有无键按下 将全部行线p1.4-p1.7置低电平,当然p1.0-p1.3为高电平(或许芯片内部已经将这些引脚它上拉),然后检测列线的状态。只要有一列的电平为低,则表示键盘中有键被按下,而且闭合的键位于低电平线与4根行线相交叉的4个按键之中。若所有列线均为高电平,则键盘中无键按下。

2、判断闭合键所在的位置 在确认有键按下后,即可进入确定具体闭合键的过程。其方法是:依次将行线置为低电平,即在置某根行线为低电平时,其它线为高电平。在确定某根行线位置为低电平后,再逐行检测各列线的电平状态。若某列为低,则该列线与置为低电平的行线交叉处的按键就是闭合的按键。

方法二:先从p1口的高四位输出低电平,低四位输出高电平,从p1口的低四位读取键盘状态。再从p1口的低四位输出低电平,高四位输出高电平,从p1口的高四位读取键盘状态。将两次读取结果组合起来就可以得到当前按键的特征编码。

矩阵键盘扫描程序

使用芯片STM8S003

所用端口:PD2~PD6, PA1~PA3

其中,PD3~PD6为输出,PA1~PA3 / PD2为输入(默认上拉)

/*

PortCom BIT7 BIT6 BIT5 BIT4 BIT3 BIT2 BIT1 BIT0

PD6 PD5 PD4 PD3 PD2 PA3 PA2 PA1

*/

程序如下:

/* 添加包含芯片的头文件 */

#include《iostm8s103f3.h》

volatile unsigned char CF[4]; //按键触发标志(表示4列,每一列同一行的

//值是一样的但列标不一样来区分不同列的键)

volatile unsigned char Cont[4];

unsigned char KeyVal; //键值

//unsigned char KeyOut[4] = {0xef,0xdf,0xbf,0x7f}; //4X4按输出端控制

//unsigned char KeyOut[4] = {0x7f,0xbf,0xdf,0xef};

unsigned char KeyOut[4] = {0x3f,0x5f,0x6f,0x77}; //两个端口组合4x4端口输出控制

unsigned char PortCom; //两个端口组合的端口

unsigned char cIn0,cIn1,cIn2,cIn3;

/*******************************************************************************

**函数名称:void delay(unsigned int ms) Name: void delay(unsigned int ms)

**功能描述:大概延时

**入口参数:unsigned int ms 输入大概延时数值

**输出:无

*******************************************************************************/

void delay(unsigned int ms)

{

unsigned int x , y;

for(x = ms; x 》 0; x--)

for(y = 1000 ; y 》 0 ; y--);

}

/*

**描述:新型4X4按键扫描程序 放在1ms-10ms中断内使用(十分稳定不需要再写消抖程序)

**备注:按键弹起时 keyVal = 0 单键按下 keyVal 有16个值,你自己程序可以针对不同值

**进行不同程序操作 keyVal单键值分别为

**0x01,0x02,0x04,0x08,

**0x11,0x12,0x14,0x18,

**0x21,0x22,0x24,0x28,

**0x31,0x32,0x34,0x38,

*/

void Key_Head()

{

unsigned char ReadData[4];

static unsigned char i;

/*

PortCom BIT7 BIT6 BIT5 BIT4 BIT3 BIT2 BIT1 BIT0

PD6 PD5 PD4 PD3 PD2 PA3 PA2 PA1

*/

cIn0 = 0;

cIn1 = 0;

cIn2 = 0;

cIn3 = 0;

if(++i》=4)i=0;

// PortCom = KeyOut[i]|0x0f; //忽略低4位

//输出扫描

PD_ODR = KeyOut[i];

//输入侦测

cIn0 = PA_IDR_bit.IDR1;

cIn1 = PA_IDR_bit.IDR2;

cIn2 = PA_IDR_bit.IDR3;

cIn3 = PD_IDR_bit.IDR2;

PortCom = (cIn3《《3) | (cIn2《《2) | (cIn1《《1) | cIn0;

ReadData[i] = (PortCom|0xf0)^0xff; //忽略高4位 取反

CF[i] = ReadData[i] & (ReadData[i] ^ Cont[i]);

Cont[i] = ReadData[i];

//输出键值

switch(CF[i])//第i列

{

case 0x08: KeyVal = ((i《《4)+8);break;

case 0x04: KeyVal = ((i《《4)+4);break;

case 0x02: KeyVal = ((i《《4)+2);break;

case 0x01: KeyVal = ((i《《4)+1);break;

default:KeyVal = 0;break;

}

delay(30);

}

/*******************************************************************************

**函数名称:void ALL_LED_Init() Name: void ALL_LED_Init()

**功能描述:初始化LED灯的IO口设为输出

**入口参数:无

**输出:无

*******************************************************************************/

void ALL_LED_Init()

{

//LED1 Init

// PD_DDR_bit.DDR2 = 1; //设置端口PD-》2的输入输出方向寄存器为输出方向

// PD_CR1_bit.C12 = 1; //设置PD2为推挽输出

// PD_CR2_bit.C22 = 1; //设置PD2的输出最大速度为10MHZ

//LED2 Init

PC_DDR_bit.DDR7 = 1; //设置端口PC-》7的输入输出方向寄存器为输出方向

PC_CR1_bit.C17 = 1; //设置PC7为推挽输出

PC_CR2_bit.C27 = 1; //设置PC7的输出最大速度为10MHZ

//LED3 Init

PC_DDR_bit.DDR6 = 1; //设置端口PC-》6的输入输出方向寄存器为输出方向

PC_CR1_bit.C16 = 1; //设置PC6为推挽输出

PC_CR2_bit.C26 = 1; //设置PC6的输出最大速度为10MHZ

//LED4 Init

PC_DDR_bit.DDR3 = 1; //设置端口PC-》3的输入输出方向寄存器为输出方向

PC_CR1_bit.C13 = 1; //设置PC3为推挽输出

PC_CR2_bit.C23 = 1; //设置PC3的输出最大速度为10MHZ

}

/*******************************************************************************

**函数名称:ALLKeyInit()

**功能描述:配置Key1 , Key2 , Key3输入按键

**入口参数:无

**输出:无

*******************************************************************************/

void ALLKeyInit()

{

//PA1_Init

PA_DDR_bit.DDR1 = 0; //GPA-》PIN3 设置为输入模式

PA_CR1_bit.C11 = 1; //GPA-》PIN3 带上拉电阻输入

PA_CR2_bit.C21 = 0; //GPA-》PIN3 禁止外部中断

//PA2_Init

PA_DDR_bit.DDR2 = 0; //GPA-》PIN3 设置为输入模式

PA_CR1_bit.C12 = 1; //GPA-》PIN3 带上拉电阻输入

PA_CR2_bit.C22 = 0; //GPA-》PIN3 禁止外部中断

//PA3_Init

PA_DDR_bit.DDR3 = 0; //GPA-》PIN3 设置为输入模式

PA_CR1_bit.C13 = 1; //GPA-》PIN3 带上拉电阻输入

PA_CR2_bit.C23 = 0; //GPA-》PIN3 禁止外部中断

//PD2_Init

PD_DDR_bit.DDR2 = 0; //GPD-》PIN3 设置为输入模式

PD_CR1_bit.C12 = 1; //GPD-》PIN3 带上拉电阻输入

PD_CR2_bit.C22 = 0; //GPD-》PIN3 禁止外部中断

//PD3_Init

PD_DDR_bit.DDR3 = 1; //GPD-》PIN3 设置为输入模式

PD_CR1_bit.C13 = 1; //GPD-》PIN3 带上拉电阻输入

PD_CR2_bit.C23 = 1; //GPD-》PIN3 禁止外部中断

//PD4_Init

PD_DDR_bit.DDR4 = 1; //GPD-》PIN3 设置为输入模式

PD_CR1_bit.C14 = 1; //GPD-》PIN3 带上拉电阻输入

PD_CR2_bit.C24 = 1; //GPD-》PIN3 禁止外部中断

//PD5_Init

PD_DDR_bit.DDR5 = 1; //GPC-》PIN5 设置为输入模式

PD_CR1_bit.C15 = 1; //GPC-》PIN5 带上拉电阻输入

PD_CR2_bit.C25 = 1; //GPC-》PIN5 禁止外部中断

//PD6_Init

PD_DDR_bit.DDR6 = 1; //GPC-》PIN5 设置为输入模式

PD_CR1_bit.C16 = 1; //GPC-》PIN5 带上拉电阻输入

PD_CR2_bit.C26 = 1; //GPC-》PIN5 禁止外部中断

}

int main(void)

{

CLK_CKDIVR = 0x00; //内部时钟为1分频

ALL_LED_Init(); //调用LED1初始化函数

ALLKeyInit(); //调用按钮初始化函数

while(1)

{

Key_Head();

switch(KeyVal)

{

case 0x01:

{

PC_ODR ^= 0x80; //异或取反LED2使其亮灭

break;

}

case 0x02:

{

PC_ODR ^= 0x40; //异或取反LED3使其亮灭

break;

}

case 0x04:

{

PC_ODR ^= 0x08; //异或取反LED4使其亮灭

break;

}

case 0x08:

{

PC_ODR ^= 0x80; //异或取反LED2使其亮灭

PC_ODR ^= 0x40; //异或取反LED3使其亮灭

break;

}

case 0x11:

{

PC_ODR ^= 0x80; //异或取反LED2使其亮灭

break;

}

case 0x12:

{

PC_ODR ^= 0x40; //异或取反LED3使其亮灭

break;

}

case 0x14:

{

PC_ODR ^= 0x08; //异或取反LED4使其亮灭

break;

}

case 0x18:

{

PC_ODR ^= 0x80; //异或取反LED2使其亮灭

PC_ODR ^= 0x40; //异或取反LED3使其亮灭

break;

}

case 0x21:

{

PC_ODR ^= 0x80; //异或取反LED2使其亮灭

break;

}

case 0x22:

{

PC_ODR ^= 0x40; //异或取反LED3使其亮灭

break;

}

case 0x24:

{

PC_ODR ^= 0x08; //异或取反LED4使其亮灭

break;

}

case 0x28:

{

PC_ODR ^= 0x80; //异或取反LED2使其亮灭

PC_ODR ^= 0x40; //异或取反LED3使其亮灭

break;

}

case 0x31:

{

PC_ODR ^= 0x80; //异或取反LED2使其亮灭

break;

}

case 0x32:

{

PC_ODR ^= 0x40; //异或取反LED3使其亮灭

break;

}

case 0x34:

{

PC_ODR ^= 0x08; //异或取反LED4使其亮灭

break;

}

case 0x38:

{

PC_ODR ^= 0x80; //异或取反LED2使其亮灭

PC_ODR ^= 0x40; //异或取反LED3使其亮灭

break;

}

default:

{

KeyVal = 0;

break;

}

}

}

}

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 矩阵键盘
    +关注

    关注

    7

    文章

    204

    浏览量

    31134
收藏 人收藏

    评论

    相关推荐

    请问51单片机中如何从矩阵键盘中分解出独立按键

    请问51单片机中如何从矩阵键盘中分解出独立按键
    发表于 11-08 06:51

    关于51单片机的矩阵键盘扫描的两种方法的困惑求解

    时间就去扫描按键,理论上和主程序的while(1)循环里代码是否长和消耗时间应该无关,应该是很可靠的一种按键扫描
    发表于 10-26 08:02

    51单片机4*4矩阵键盘有什么好方法扫描识别

    关于51单片机的4*4矩阵键盘,有什么你比较好扫描识别方法,麻烦告知一下,十分感谢!!
    发表于 10-24 06:23

    矩阵键盘在进行按键检测的时候怎么识别长按?

    矩阵键盘在进行按键检测的时候怎么识别长按
    发表于 10-12 06:01

    分享一个stm32矩阵键盘程序

    分享一个stm32矩阵键盘程序
    发表于 10-10 08:26

    stm32矩阵键盘程序

    电子发烧友网站提供《stm32矩阵键盘程序.zip》资料免费下载
    发表于 10-07 15:07 9次下载
    stm32<b class='flag-5'>矩阵</b><b class='flag-5'>键盘</b>的<b class='flag-5'>程序</b>

    利用单片机实现数码管显示4乘4键盘矩阵按键的C语言程序

    使用单片机实现数码管显示4乘4键盘矩阵按键的C语言程序
    发表于 09-20 08:29

    在Proteus软件中如何读取矩阵键盘

    矩阵键盘是读取一系列按键的有效方法,例如读取电话的拨号盘、POS机小键盘、电脑键盘,甚至是电子钢
    的头像 发表于 07-19 15:26 2519次阅读
    在Proteus软件中如何读取<b class='flag-5'>矩阵</b><b class='flag-5'>键盘</b>?

    今天看一下矩阵键盘怎么用。

    矩阵键盘
    YS YYDS
    发布于 :2023年07月02日 18:08:06

    20键键盘矩阵开源分享

    电子发烧友网站提供《20键键盘矩阵开源分享.zip》资料免费下载
    发表于 06-12 16:04 0次下载
    20键<b class='flag-5'>键盘</b><b class='flag-5'>矩阵</b>开源分享

    51单片机矩阵键盘扫描程序

    51单片机矩阵键盘扫描程序
    发表于 05-17 17:20 4次下载

    基于89C51单片机的矩阵键盘反转扫描程序

    基于89C51单片机的矩阵键盘反转扫描程序
    发表于 05-15 11:02 2次下载

    基于89C51单片机的矩阵键盘行列扫描程序

    基于89C51单片机的矩阵键盘行列扫描程序
    发表于 05-15 11:00 1次下载

    基于89C51单片机的矩阵键盘中断扫描程序

    基于89C51单片机的矩阵键盘中断扫描程序
    发表于 05-15 11:00 2次下载

    基于ATmega8单片机4×4矩阵键盘识别Proteus仿真源程序

    基于ATmega8单片机4×4 矩阵键盘识别Proteus仿真源程序
    发表于 05-05 09:44 0次下载