矩阵键盘程序

矩阵键盘程序,第1张

假设按下的是S1键进行如下检测(4*4键盘

先在P3口输出

p3 00001111

低四位 行会有变化

cord_h =00001111&00001110 =00001110

if !=00001111

延时0.1us

cord_h=00001110&00001111=00001110

if !=00001111

P3再输出11111110

P3=00001110|11110000=11111110

输出高四位

cord_l=P3&0xf0 //简答磨此时P3口就是输入值01111110 而不是上面的11111110

cord_l=01111110&11110000=01110000

cord_h+cord_l=00001110+01110000=01111110=0x7e //此编码即为S1的编码

#include <reg52.h>//包含头文件

#define uchar unsigned char

#define uint unsigned int

unsigned char const table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,

0x77,0x7c,0x39,0x5e,0x79,0x71}//0-F

uchar keyscan(void)

void delay(uint i)

void main()

{

uchar key

P2=0x00//1数码管亮 按相应的按键,会显示按键上的字符

while(1)

{

key=keyscan()//调用键盘扫描,

switch(key)

{

case 0x7e:P0=table[0]break//0 按下相应的键显示相对应的码值

case 0x7d:P0=table[1]break//1

case 0x7b:P0=table[2]break//2

case 0x77:P0=table[3]break//3

case 0xbe:P0=table[4]break//4

case 0xbd:P0=table[5]break//5

case 0xbb:P0=table[6]break//6

case 0xb7:P0=table[7]break//7

case 0xde:P0=table[8]break//8

case 0xdd:P0=table[9]break//9

case 0xdb:P0=table[10]break//a

case 0xd7:P0=table[11]break//b

case 0xee:P0=table[12]break//c

case 0xed:P0=table[13]break//d

case 0xeb:P0=table[14]break//e

case 0xe7:P0=table[15]break//f

}

}

}

uchar keyscan(void)//键盘扫描函数,使用行列反转扫描法

{

uchar cord_h,cord_l//行列值

P3=0x0f //行线输出全为0

cord_h=P3&0x0f//读入列线值

if(cord_h!=0x0f)/举拍/先检测有无按键按下

{

delay(100) //去抖

cord_h=P3&0x0f //拦斗读入列线值

if(cord_h!=0x0f)

{

P3=cord_h|0xf0 //输出当前列线值

cord_l=P3&0xf0 //读入行线值

return(cord_h+cord_l)//键盘最后组合码值

}

}return(0xff)//返回该值

}

void delay(uint i)//延时函数

{

while(i--)

}

#include <reg52.h>

#define key P2

int val

char code sign[17]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x83,0xf8,0x80,0x98,0x88,0x83,0xc6,0xa1,0x86,0x8e}

void delay_10ms()

void key_scan() //矩阵按键函数声明

void main()

{

while(1)

{

P0=sign[val]

key_scan()

}

}

void key_scan()

{

static char a,b //a用于判断是不是第一次按下,b用于稍微延时

char state //记录键值

key=0x0f

if(key!=0x0f) //如果有按键按下

{

state=key //记录第一次按键状态

if(a==0)//如果第一次按下

{

b++//略微延时 不需禅穗要延时10ms以提高单片机效率

if(b>3)

{

if(key==state)//判断键值做指是否和第一次一致贺胡卜

{

a=1,b=0 //复位状态值 a==1说明已经按下,等待后续松开

key=0x0f//测试列

switch(key)

{

case (0x07): val=0break

case (0x0b): val=1break

case (0x0d): val=2break

case (0x0e): val=3break

}

key=0xf0 // 测试行

switch(key)

{

case (0x70): val=valbreak

case (0xb0): val=val+4break

case (0xd0):val=val+8break

case (0xe0): val=val+12break

}

}

}

}

}

else

{

a=b=0//松开后复位状态机

}

}

4x4键盘的程序有扫描法与线反法,但我个人认为用线反法较好,用扫描法得依次扫描所有行或列,如果用线反法就简单多了。先使键盘的行置为低、列置为高(或列置为高、行置为低),接着读回端口的值。比如:如果使用P0为键盘接口就先使低四位为低、高四位为高即P0=0xf0然后就读回P0口的值赋给一个变量,a=P0紧接就给行列赋相反的值行置为高、列置为低(或列置为低、行置为高)即P0=0x0f然后就读回再与a运算就能得到唯一的识别码下面的程序就是用线反写一个4x4键盘识别程序:#include<AT89X52.H>

#include<delay.h>#define KEY_SCAN P1

#define uchar unsigned char//char num/********************************/

/*函数名称:KEY_DOWN() */

/*函数功能:延时子函数 */

/*参数:无 */

/*返回:返回1或0*/

/*备注:1表示有键按下,0则无*/

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

bit KEY_DOWN()

{

KEY_SCAN=0x0f //先给键盘口赋个初值

if(KEY_SCAN!=0x0f) //没肆判断是有按键按下,即KEY_SCAN不等于初值时有键按下

{

delayms(10)//消抖

if(KEY_SCAN!=0x0f) //再次判断是否真有键按下

return 1 //真有就返回1没有返回零

else

return 0

}

else

return 0

}/********************************/

/*函数名称:SCAN_GET() */

/*函数功能:键盘值函数 */

/*参数:无 */

/*返回:返回1或0*/

/*备注:无 */

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

uchar SCAN_GET()

{

char button

uchar key_code

button=KEY_SCAN

KEY_SCAN=0xf0

button=(button|KEY_SCAN)

while(KEY_SCAN!=0xf0)

delayms(10)

switch(button)

{

case 0xd7: key_code='1'break

case 0xdb: key_code='2'break

case 0xdd: key_code='3'break

case 0xb7: key_code='4'break

case 0xbb: key_code='5'break

case 0xbd: key_code='6'break

case 0x77: key_code='7'break

case 0x7b: key_code='8'break

case 0x7d: key_code='9'break

case 0xeb: key_code='0'break

case 0xee: key_code=0xeebreak

default : break

}

return key_code

}

//////////////////////////////////////////////////////////////

//此程序是上两个程序结合的

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

/*函数名称:亩郑Key_Get() */

/*函数功能:键盘扫描函数*/

/*参数:无 */

/*返回:无 */

/*备迅察颂注:无 */

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

void Key_Get()

{

char button

KEY_SCAN=0x0f

if(KEY_SCAN!=0x0f)

{

delayms(5)

if(KEY_SCAN!=0x0f)

{

button=KEY_SCAN

KEY_SCAN=0xf0

button=(button|KEY_SCAN)

while(KEY_SCAN!=0xf0)

switch(button)

{

case 0xd7: num='1'P0=0x00break

case 0xdb: num='2'P0=0x0fbreak

case 0xdd: num='3'break

case 0xb7: num='4'break

case 0xbb: num='5'break

case 0xbd: num='6'break

case 0x77: num='7'break

case 0x7b: num='8'break

case 0x7d: num='9'break

case 0xeb: num='0'break

case 0xe7: num='a'break

case 0xed: num='b'break

case 0xee: num='c'break

case 0xde: num='d'break

case 0xbe: num='e'break

case 0x7e: num='f'break

default : break

}

}

}

}


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/12217083.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-21
下一篇 2023-05-21

发表评论

登录后才能评论

评论列表(0条)

保存