51单片机,用c语言,如何用矩阵键盘为一个变量char a赋值?已设计出一个函数,使得每按一个矩阵

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了51单片机,用c语言,如何用矩阵键盘为一个变量char a赋值?已设计出一个函数,使得每按一个矩阵相关的知识,希望对你有一定的参考价值。

51单片机,用c语言,如何用矩阵键盘为一个变量char a赋值?已设计出一个函数,使得每按一个矩阵键盘的键都return一个数

/***4×4矩阵按键构架——火柴天堂作品-20110921***/
/***源程序默认硬件环境:52单片机,12MHz晶振,P1口 4×4矩阵键盘,详细布局如下***/
/* 默认矩阵布局,按键扫描方式1使用
P0 P1 P2 P3
│ │ │ │
P4─┼──┼──┼──┤ S1 S2 S3 S4
│ │ │ │
P5─┼──┼──┼──┤ S5 S6 S7 S8
│ │ │ │
P6─┼──┼──┼──┤ S9 S10 S11 S12
│ │ │ │
P7─┴──┴──┴──┘ S13 S14 S15 S16
*/

#include"reg52.h" //包含52头文件
#define TRUE 1 //定义布尔量\'1\':真
#define FALSE 0 //定义布尔量\'0\':假
#define uchar unsigned char //定义 无符号字符型数据 简称
#define uint unsigned int //定义 无符号整型数据 简称
#define KeyPort P1
#define Key1Value 1
#define Key2Value 2

//uchar code KeyCodeList[16]=0xee,0xed,0xeb,0xe7,0xde,0xdd,0xeb,0xe7,0xbe,0xbd,0xbb,0xb7,0x7e,0x7d,0x7b,0x77;//按键代码列表,按键扫描方式2使用,可按需要随意修改顺序

uchar KeyScan() //按键扫描函数(方式1,需配合源程序矩阵布局,返回值0表示无按键,1-16为对应按键),缺点:无法扫描组合键(同时按2个按键以上)

uchar temp_h,temp_l,scan_value,i;
KeyPort=0xf0; //设置低4位为0(扫描线),准备读取高4位(返回线)状态
if(KeyPort==0xf0) return 0; //若高4位状态不变,表示无按键,返回无按键 键值0
temp_h=~KeyPort>>4; //若高4位状态改变,表示有按键,读取高4位,并将结果转成正逻辑(按键对应 行线 为 1 )存在temp_h低位上
KeyPort=0x0f; //设置高4位为0(扫描线),准备读取低4位(返回线)状态
temp_l=~(KeyPort|0xf0); //读取低4位,并将结果转成正逻辑(按键对应 列线 为 1 )存在temp_l低位上
while(i<4) //将 按键行线号 转成数值

if((temp_h>>i)==0x01) break; //读取按键行线号
i++;

if(i==4) return 0; //若读取出错,返回无按键
temp_h=i; //将 按键行线数值 结果存于 temp_h
i=0;
while(i<4) //将 按键列线号 转成数值

if((temp_l>>i)==0x01) break; //读取按键列线号
i++;

if(i==4) return 0; //若读取出错,返回无按键
temp_l=i; //将 按键列线号 结果存于 temp_l
scan_value=(temp_h<<2)+temp_l+1; //合并行列线数值,并转成按键值,每条行线键值差为 4(temp_h<<2),按键值从1开始(+1,0为无按键 键值)
return scan_value; //返回 按键值

/*
uchar KeyScan() //按键扫描函数(方式2,需配合 按键代码列表 数组 进行 键值 查询,返回值0表示无按键,1-16为代码表顺序对应按键),缺点:无法扫描组合键

uchar temp,i;
KeyPort=0xf0; //设置低4位为0(扫描线),准备读取高4位(返回线)状态
if(KeyPort==0xf0) return 0; //若高4位状态不变,表示无按键,返回无按键 键值0
temp=KeyPort|0x0f; //若高4位状态改变,表示有按键,读取高4位,并将结果存于 temp 的高4位
KeyPort=0x0f; //设置高4位为0(扫描线),准备读取低4位(返回线)状态
temp&=(KeyPort|0xf0); //读取低4位,并将结果存于 temp 的低4位
while(i<16) //将按键行列线代码转换成键值

if(temp==KeyCodeList[i]) break; //将 行列线 代码与 按键代码表 进行对比,若一致则结束对比
i++; //进行下一个对比

if(i==16) return 0; //若查询出错,或 行列线代码 不在 按键代码表中,返回无按键 键值0
return i+1; //返回按键值1~16(按键代码列表中元素下标+1,0为无按键 键值)

*/
void KeyResp() //按键响应函数

static uchar KeyValue; //定义静态变量-按键值,
static bit KeyDownFlag,KeyReadyFlag; //定义静态标志位-按键按下标志,按键准备(响应)标志
uchar key_value=KeyScan(); //调用扫描函数,并将结果临时存放于key_value 中
if(key_value) //若扫描结果为真(即有按键)

KeyValue=key_value; //保存扫描结果
KeyDownFlag=TRUE; //按键按下标志 置位
KeyReadyFlag=TRUE; //按键准备(响应)标志 置位

else KeyDownFlag=FALSE; //若扫描结果为假(即无按键),则清空按键按下标志
if(KeyReadyFlag && !KeyDownFlag) //若 按键已准备(响应),且无按键按下,(可知为 按下后又松手情况)

switch(KeyValue) //查找 按键值 对应的 按键处理

case Key1Value:break; //按键1处理
case Key2Value:break; //按键2处理
//case Key3Value:break;//......... //按键N处理
default:break; //无对应按键,或其它按键值处理

KeyValue=0; //清除按键值
KeyReadyFlag=FALSE; //清空 按键准备(响应)标志



void main() //主函数

while(1) //循环系统

KeyResp(); //调用 按键响应函数

参考技术A 首先给矩阵键盘的各个按键编码(键码),然后扫描键盘,检测到按键按下用return返回键码。追问

已经编好这个子函数。我想用这个来调时钟,下一步怎么做?

追答

是这样,我不太有时间仔细看你的源文件,告诉你个方法吧。刚才已经解决了键码的问题,键码是给单片机用的,让它知道哪个键按下了。现在你还得给每个键取个名字(在心里也行),这是给你用的,让你知道那个键干什么。可以命名年加、年减、月加、月减、日加、日减、小时加、小时减、分钟加、分钟减、秒加、秒减这些键,这样按键键码和名字就一一对应了,检测到按键时,用条件判断语句判断键码对应执行相应变量的加或者减。

追问

我是想直接输入数字调时间,而不是通过加减,有大概思路吗?谢谢?

追答

也可以实现,方法和上面差不多,只不过是不加减直接修改变量。16个按键,去掉10个数字键,还有6个供你定义(你总得自己明白也得让单片机知道做什么),刚好够用。比方你定义分钟调整键,按下时让分钟那两位数闪烁,愿意的话就闪烁“--”,然后按数字键,按第一个修改十位数,按第二个修改个位数,以此类推。

追问

我的问题就在这。我想不到用怎么样的算法能让程序停在某一个位置,当我按下一个数字后,程序才继续执行

追答

按键扫描部分你搞好了吗?

追问

搞好了

追答

已经有人回答了,你参考下。

单片机告诉一个地址,怎样用C语言编程读取地址处的值???

例如: 4X4键盘 物理地址为01F700H怎样用C语言实现对键盘进行循环扫描???

参考一下吧:
#include<reg52.h> //包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义

#define DataPort P0 //定义数据端口 程序中遇到DataPort 则用P0 替换
#define KeyPort P1

sbit LATCH1=P2^2;//定义锁存使能端口 段锁存
sbit LATCH2=P2^3;// 位锁存

unsigned char code dofly_DuanMa[]=0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,
0x77,0x7c,0x39,0x5e,0x79,0x71;// 显示段码值0~F
unsigned char code dofly_WeiMa[]=0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f;//分别对应相应的数码管点亮,即位码
unsigned char TempData[8]; //存储显示值的全局变量

void DelayUs2x(unsigned char t);//us级延时函数声明
void DelayMs(unsigned char t); //ms级延时
void Display(unsigned char FirstBit,unsigned char Num);//数码管显示函数
unsigned char KeyScan(void);//键盘扫描
unsigned char KeyPro(void);
void Init_Timer0(void);//定时器初始化
/*------------------------------------------------
主函数
------------------------------------------------*/
void main (void)

unsigned char num,i,j;
unsigned char temp[8];
Init_Timer0();

while (1) //主循环


num=KeyPro();
if(num!=0xff)

if(i<8)

temp[i]=dofly_DuanMa[num];
for(j=0;j<=i;j++)
TempData[7-i+j]=temp[j];

i++;
if(i==9)//多出一个按键输入为了清屏 原本应该为8

i=0;
for(j=0;j<8;j++)//清屏
TempData[j]=0;


//Display(0,8); //显示全部8位
//主循环中添加其他需要一直工作的程序



/*------------------------------------------------
uS延时函数,含有输入参数 unsigned char t,无返回值
unsigned char 是定义无符号字符变量,其值的范围是
0~255 这里使用晶振12M,精确延时请使用汇编,大致延时
长度如下 T=tx2+5 uS
------------------------------------------------*/
void DelayUs2x(unsigned char t)

while(--t);

/*------------------------------------------------
mS延时函数,含有输入参数 unsigned char t,无返回值
unsigned char 是定义无符号字符变量,其值的范围是
0~255 这里使用晶振12M,精确延时请使用汇编
------------------------------------------------*/
void DelayMs(unsigned char t)


while(t--)

//大致延时1mS
DelayUs2x(245);
DelayUs2x(245);


/*------------------------------------------------
显示函数,用于动态扫描数码管
输入参数 FirstBit 表示需要显示的第一位,如赋值2表示从第三个数码管开始显示
如输入0表示从第一个显示。
Num表示需要显示的位数,如需要显示99两位数值则该值输入2
------------------------------------------------*/
void Display(unsigned char FirstBit,unsigned char Num)

static unsigned char i=0;

DataPort=0; //清空数据,防止有交替重影
LATCH1=1; //段锁存
LATCH1=0;

DataPort=dofly_WeiMa[i+FirstBit]; //取位码
LATCH2=1; //位锁存
LATCH2=0;

DataPort=TempData[i]; //取显示数据,段码
LATCH1=1; //段锁存
LATCH1=0;

i++;
if(i==Num)
i=0;


/*------------------------------------------------
定时器初始化子程序
------------------------------------------------*/
void Init_Timer0(void)

TMOD |= 0x01; //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
//TH0=0x00; //给定初值
//TL0=0x00;
EA=1; //总中断打开
ET0=1; //定时器中断打开
TR0=1; //定时器开关打开

/*------------------------------------------------
定时器中断子程序
------------------------------------------------*/
void Timer0_isr(void) interrupt 1

TH0=(65536-2000)/256; //重新赋值 2ms
TL0=(65536-2000)%256;

Display(0,8); // 调用数码管扫描



/*------------------------------------------------
按键扫描函数,返回扫描键值
------------------------------------------------*/
unsigned char KeyScan(void) //键盘扫描函数,使用行列逐级扫描法

unsigned char Val;
KeyPort=0xf0;//高四位置高,低四位拉低
if(KeyPort!=0xf0)//表示有按键按下

DelayMs(10); //去抖
if(KeyPort!=0xf0)
//表示有按键按下
KeyPort=0xfe; //检测第一行
if(KeyPort!=0xfe)

Val=KeyPort&0xf0;
Val+=0x0e;
while(KeyPort!=0xfe);
DelayMs(10); //去抖
while(KeyPort!=0xfe);
return Val;

KeyPort=0xfd; //检测第二行
if(KeyPort!=0xfd)

Val=KeyPort&0xf0;
Val+=0x0d;
while(KeyPort!=0xfd);
DelayMs(10); //去抖
while(KeyPort!=0xfd);
return Val;

KeyPort=0xfb; //检测第三行
if(KeyPort!=0xfb)

Val=KeyPort&0xf0;
Val+=0x0b;
while(KeyPort!=0xfb);
DelayMs(10); //去抖
while(KeyPort!=0xfb);
return Val;

KeyPort=0xf7; //检测第四行
if(KeyPort!=0xf7)

Val=KeyPort&0xf0;
Val+=0x07;
while(KeyPort!=0xf7);
DelayMs(10); //去抖
while(KeyPort!=0xf7);
return Val;



return 0xff;

/*------------------------------------------------
按键值处理函数,返回扫键值
------------------------------------------------*/
unsigned char KeyPro(void)

switch(KeyScan())

case 0x7e:return 0;break;//0 按下相应的键显示相对应的码值
case 0x7d:return 1;break;//1
case 0x7b:return 2;break;//2
case 0x77:return 3;break;//3
case 0xbe:return 4;break;//4
case 0xbd:return 5;break;//5
case 0xbb:return 6;break;//6
case 0xb7:return 7;break;//7
case 0xde:return 8;break;//8
case 0xdd:return 9;break;//9
case 0xdb:return 10;break;//a
case 0xd7:return 11;break;//b
case 0xee:return 12;break;//c
case 0xed:return 13;break;//d
case 0xeb:return 14;break;//e
case 0xe7:return 15;break;//f
default:return 0xff;break;

参考技术A 4X4键盘还有物理地址?
你是怎么得到这个物理地址的呢?
-待续-

以上是关于51单片机,用c语言,如何用矩阵键盘为一个变量char a赋值?已设计出一个函数,使得每按一个矩阵的主要内容,如果未能解决你的问题,请参考以下文章

51单片机C语言,电子密码的验证输入程序,矩阵键盘用作输入,密码正确时,向串口送1,否则送0

求。用89C51单片机设计4*4矩阵式键盘计算器程序

基于51单片机LCD1602矩阵键盘电子密码锁proteus仿真设计

怎么用51单片机4*4键盘实现简单计算器 就要加减就可以 汇编语言 麻烦可

如何用C语言编写一个LED灯渐变的程序

如何用switch语句判断从键盘输入字符的属性属于大写字母,小写字母,数字或其他的符号