矩阵键盘扫描程序分析

#include<reg52.h>//包含头文件#defineucharunsignedchar#defineuintunsignedintunsignedcharcon... #include <reg52.h>//包含头文件
#define uchar unsigned char
#define uint unsigned int
unsigned char const dofly[]={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=dofly[0];break;//0 按下相应的键显示相对应的码值 原理就是高四位一列低四位一列的组
//合。0111 1110 7e 0表示按键后为0,1表示没有按键按下的。即P3.7与P3.1连接为低电平,为S1键
//其他类推。
case 0x7d:P0=dofly[1];break;//1
case 0x7b:P0=dofly[2];break;//2
case 0x77:P0=dofly[3];break;//3
case 0xbe:P0=dofly[4];break;//4
case 0xbd:P0=dofly[5];break;//5
case 0xbb:P0=dofly[6];break;//6
case 0xb7:P0=dofly[7];break;//7
case 0xde:P0=dofly[8];break;//8
case 0xdd:P0=dofly[9];break;//9
case 0xdb:P0=dofly[10];break;//a
case 0xd7:P0=dofly[11];break;//b
case 0xee:P0=dofly[12];break;//c
case 0xed:P0=dofly[13];break;//d
case 0xeb:P0=dofly[14];break;//e
case 0xe7:P0=dofly[15];break;//f
}
}
}
uchar keyscan(void)//键盘扫描函数,使用行列反转扫描法 比如:行为低电位,列为高四位
{
uchar cord_h,cord_l;//行列值
P3=0x0f; //行线输出全为0
cord_h=P3&0x0f; //读入列线值
if(cord_h!=0x0f) //先检测有无按键按下
{
delay(100); //去抖
if(cord_h!=0x0f)
{
cord_h=P3&0x0f; //读入列线值
P3=cord_h|0xf0; //输出当前列线值
cord_l=P3&0xf0; //读入行线值
return(cord_h+cord_l);//键盘最后组合码值
}
}return(0xff); //返回该值
}
void delay(uint i)//延时函数
{
while(i--);
}

主要是键盘扫描这个小程序不懂,请各位帮忙解释解释~~~
if(cord_h!=0x0f)
{
cord_h=P3&0x0f; //读入列线值
P3=cord_h|0xf0; //输出当前列线值
cord_l=P3&0xf0; //读入行线值
return(cord_h+cord_l);//键盘最后组合码值
}
}return(0xff); //返回该值
假设P3=7e,那执行这段程序后:
cord_h=0x0e,
P3=0xfe,
cord_l=0xf0,
return(0x0e+0xf0);键盘的最后组合码值是什么?是0xfe么?如果是,case里也没这个值呀?是吧?我这样算对么?
展开
 我来答
glbdwh
2011-03-04 · TA获得超过601个赞
知道答主
回答量:59
采纳率:0%
帮助的人:0
展开全部
假设按下的是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--);
}
追问
cord_l=P3&0xf0 //此时P3口就是输入值01111110 而不是上面的11111110
为什么此时的P3不是fe?它的上个语句不是已经改变P3的值了么?
happyming0809
2011-03-04 · TA获得超过757个赞
知道小有建树答主
回答量:624
采纳率:0%
帮助的人:327万
展开全部
uchar keyscan(void)//键盘扫描函数,使用行列反转扫描法 比如:行为低电位,列为高四位
{
uchar cord_h,cord_l;//行列值
P3=0x0f; //行线输出全为0
cord_h=P3&0x0f; //读入列线值
if(cord_h!=0x0f) //这个句话意思是这样的,如果有按键按下的话。那肯定就不是0x0f了。
{ 主要就是识别按键按下作用
delay(100); //去抖
if(cord_h!=0x0f) //这是双重判定作用而已
{
cord_h=P3&0x0f; //读入列线值
P3=cord_h|0xf0; //输出当前列线值
cord_l=P3&0xf0; //读入行线值
return(cord_h+cord_l);//键盘最后组合码值
}
}return(0xff); //返回该值
}
好像你的程序解释得很好了。但是好像这么快,最好要延时下才准的。
我教你按键的原理吧
就是那根线如果是高电平的话,遇到底电平就会变为底电平的。所以如果有按键按下的话,高电平就会变为低电平了。所以就可以读取了。
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
开心红茶杯
2011-03-05 · TA获得超过2264个赞
知道小有建树答主
回答量:628
采纳率:0%
帮助的人:408万
展开全部
下的是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口做的键盘
你的去抖检测没有做好
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
wang132409
2013-02-02
知道答主
回答量:2
采纳率:0%
帮助的人:3055
展开全部
你这需要松手检测吗
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
收起 更多回答(2)
推荐律师服务: 若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询

为你推荐:

下载百度知道APP,抢鲜体验
使用百度知道APP,立即抢鲜体验。你的手机镜头里或许有别人想知道的答案。
扫描二维码下载
×

类别

我们会通过消息、邮箱等方式尽快将举报结果通知您。

说明

0/200

提交
取消

辅 助

模 式