你可以尝试使用switch case语句代替if语句,这样可以避免程序出现参差上的错误(不用套那么多的if)
你可以尝试把key置零总觉得你用table11有点不大好
我也是新手,希望能帮到你
我给你一个我自己写的吧 有注释
假设按下的是S1键进行如下检测(44键盘)
先在P3口输出
p3 00001111
低四位 行会有变化
cord_h =00001111&00001110 =00001110
if !=00001111
延时01us
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 <reg52h>//包含头文件
#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口做的键盘
你的去抖检测没有做好
44 矩阵键盘布局如下,检测按键,然后通过 LCD1602 显示出来
第一行:The key value is
第二行:每按一次键,键值依次显示出来,整行显示完后,清屏,键值
从左开始显示
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
0png
单片机源程序如下:
/
LCD1602液晶显示实验
实现现象:下载程序后插上LCD1602液晶在开发板上,即可显示
注意事项:如果不想让点阵模块显示,可以将74HC595模块上的JP595短接片拔掉。
/
#include "reg52h" //此文件中定义了单片机的一些特殊功能寄存器
#include "lcdh"
typedef unsigned int u16; //对数据类型进行声明定义
typedef unsigned char u8;
#define LCD1602_DATAPINS P0
#define GPIO_KEY P1
u8 KeyValue,k=0,n=0,miss=1;
u8 Disp[]="The key value is";
u8 cmy[17]={'1','2','3','/','4','5','6','%','7','8','9','@','','0','#','='};
u8 dp[17]={' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' '};
/
函 数 名 : main
函数功能 : 主函数
输 入 : 无
输 出 : 无
/
void KeyDown(void)
{
char a=0;
GPIO_KEY=0x0f;
if(GPIO_KEY!=0x0f)//读取按键是否按下
{
Lcd1602_Delay1ms(10);//延时10ms进行消抖
if(GPIO_KEY!=0x0f)//再次检测键盘是否按下
{ n=1;
//测试列
GPIO_KEY=0X0F;
switch(GPIO_KEY)
{
case(0X07): KeyValue=0;break;
case(0X0b): KeyValue=1;break;
case(0X0d): KeyValue=2;break;
case(0X0e): KeyValue=3;break;
}
//测试行
GPIO_KEY=0XF0;
switch(GPIO_KEY)
{
case(0X70): KeyValue=KeyValue;break;
case(0Xb0): KeyValue=KeyValue+4;break;
case(0Xd0): KeyValue=KeyValue+8;break;
case(0Xe0): KeyValue=KeyValue+12;break;
}
while((a<50)&&(GPIO_KEY!=0xf0)) //检测按键松手检测
{
Lcd1602_Delay1ms(1);
a++;
}
}
}
}
void main(void)
{
u8 i,m;
LcdInit();
for(i=0;i<16;i++)
{
LcdWriteData(Disp[i]);
}
while(miss)
{
LcdWriteCom(0x80+0x40);
KeyDown();
if(n==1)
{dp[k]=cmy[KeyValue];
n=0;
k++;
}
for(m=0;m<16;m++)
{LcdWriteData(dp[m]);}
if(k==16)
{ for(m=0;m<16;m++)
{dp[m]=' ';}
Lcd1602_Delay1ms(1000);
LcdWriteCom(0x01);
Lcd1602_Delay1ms(1000);
k=0;
for(i=0;i<16;i++)
{
LcdWriteData(Disp[i]);
}
}
}
}
以上就是关于普中51单片机 矩阵键盘全部的内容,包括:普中51单片机 矩阵键盘、51单片机矩阵键盘扫描程序分析、51单片机4×4矩阵键盘仿真哪里找等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)