求助51按键扫描程序
下面这段程序不知道为什么在P3有按键按下时,P35,P36,P37都为0P32P33P34为列P35P36P37为行ucharvKeyScan(void){ucharuc...
下面这段程序不知道为什么在P3有按键按下时,P35,P36,P37都为0
P32 P33 P34为列
P35 P36 P37为行
uchar vKeyScan(void)
{
uchar ucKeyRow = 0;
uchar ucKeyLine= 0;
uchar ucKeyNum = 0;
P3 = P3&0xE3; /* set line=0,row=1 *//* set P35 P36 P37=1 */
ucKeyRow = P3&0xE0;
if(ucKeyRow!= 0xE0)
{
for(ucKeyNum = 0;ucKeyNum<255;ucKeyNum++);//delay some time
for(ucKeyNum = 0;ucKeyNum<255;ucKeyNum++);//delay some time
}
ucKeyRow = P3&0xE0;
if(ucKeyRow!= 0xE0)
{
P3 = (P3|0xFC)&0x1F;/* set line=1,row=0 */;/* set P32 P33 P34=1 */
ucKeyLine = P3&0x1C;这个地方任意key按下后P35,P36,P37就都为0了,不知道为啥
for(ucKeyNum = 0; ucKeyNum < sizeof(ucKEY_BOARD);ucKeyNum++)
{
if(ucKEY_BOARD[ucKeyNum] == (ucKeyLine+ucKeyRow))
{
ucReadData = ucKeyNum++;
break;
}
}
}
return ucReadData;
} 展开
P32 P33 P34为列
P35 P36 P37为行
uchar vKeyScan(void)
{
uchar ucKeyRow = 0;
uchar ucKeyLine= 0;
uchar ucKeyNum = 0;
P3 = P3&0xE3; /* set line=0,row=1 *//* set P35 P36 P37=1 */
ucKeyRow = P3&0xE0;
if(ucKeyRow!= 0xE0)
{
for(ucKeyNum = 0;ucKeyNum<255;ucKeyNum++);//delay some time
for(ucKeyNum = 0;ucKeyNum<255;ucKeyNum++);//delay some time
}
ucKeyRow = P3&0xE0;
if(ucKeyRow!= 0xE0)
{
P3 = (P3|0xFC)&0x1F;/* set line=1,row=0 */;/* set P32 P33 P34=1 */
ucKeyLine = P3&0x1C;这个地方任意key按下后P35,P36,P37就都为0了,不知道为啥
for(ucKeyNum = 0; ucKeyNum < sizeof(ucKEY_BOARD);ucKeyNum++)
{
if(ucKEY_BOARD[ucKeyNum] == (ucKeyLine+ucKeyRow))
{
ucReadData = ucKeyNum++;
break;
}
}
}
return ucReadData;
} 展开
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--);
}
在P3口做的键盘
你的去抖检测没有做好
假设按下的是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--);
}
在P3口做的键盘
你的去抖检测没有做好
追问
就是照这个程序写的,移植到我的板子上,在有键按下时,类似于cord_l=P3&0xf0; //读入行线值
ucKeyLine = P3&0x1C;这个地方任意key按下后P35,P36,P37就都为0了,只能读到key压下的列,行的值全为0
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询