一.独立按键实验
1.电路原理图
2.说明:使用排线连接JP10的和JP12,JP8连接JP5。本实验通过八个独立按键控制八个LED小灯。按键时会抖动,所以要消抖。
3.消抖检测
- 先给按键对应的I/O口置1;
- 判断I/O口是否为1;
- 若是1,则说明按键按下,若是0,则按键没有按下;
4.源代码如下:
/*********************************************** > File Name: 独立按键实验 > Author: pengshp > Mail: [email protected] > Date: 2015年 7 月 16 日*************************************************/#include <reg51.h> //此文件中定义了51的一些特殊功能寄存器#include <intrins.h>//--定义要使用的IO口--//#define GPIO_KEY P1 //独立键盘用P1口#define GPIO_LED P0 //led使用P0口void Delay10ms(unsigned int c); //延时10msunsigned char Key_Scan();void main(){ unsigned char ledValue, keyNum; ledValue = 0x01; while (1) { keyNum = Key_Scan(); //扫描键盘 switch (keyNum) { case(0xFE) : //返回按键K1的数据 ledValue = 0x01; break; case(0xFD) : //返回按键K2的数据 ledValue = 0x02; break; case(0xFB) : //返回按键K3的数据 ledValue = 0x04; break; case(0xF7) : //返回按键K4的数据 ledValue = 0x08; break; case(0xEF) : //返回按键K5的数据 ledValue = 0x10; break; case(0xDF) : //返回按键K6的数据 ledValue = 0x20; break; case(0xBF) : //返回按键K7的数据 ledValue = 0x40; break; case(0x7F) : //返回按键K8的数据 ledValue = 0x80; break; default: break; } GPIO_LED = ledValue; //点亮LED灯 } }unsigned char Key_Scan() //扫描键盘{ unsigned char keyValue = 0 , i; //保存键值 //--检测按键1--// if (GPIO_KEY != 0xFF) //检测按键K1是否按下 { Delay10ms(1); //消除抖动 if (GPIO_KEY != 0xFF) //再次检测按键是否按下 { keyValue = GPIO_KEY; i = 0; while ((i<50) && (GPIO_KEY != 0xFF)) //检测按键是否松开 { Delay10ms(1); i++; } } } return keyValue; //将读取到键值的值返回}void Delay10ms(unsigned int c) //误差 0us{ unsigned char a, b; for (;c>0;c--) { for (b=38;b>0;b--) { for (a=130;a>0;a--); } } }
二.矩阵键盘实验
1.电路原理图
2.矩阵键盘原理:矩阵键盘一端接行线,一端接列线,行线控制高四位,列线控制低四位;每个按键位于行线和列线的交叉点。
3.矩阵键盘扫描
(1).逐行扫描
- 高四位轮流输出低电平,当低四位接受的数据不全为0时,则按键按下,并可判断哪个按键按下。
(2).行列扫描
- 高四位全部输出低电平,低四位全部输出高电平,当接受的数据低四位不全为高电平时,说明有按键按下;然后反过来,高四位输出高电平,低四位输出低电平,根据接收到的高四位的值来判断哪一行有按键按下。
4.说明:可通过按下按键并在液晶屏上显示相应的键值。但较为复杂,液晶屏后面介绍。接线为JP4接JP8(P1),JP10接JP12,并将JP165断开。
5.程序源代码
/************************************** > File Name: 矩阵键盘实验 > Author: pengshp > Mail: [email protected] > Date: 2015年 7 月 16 日***************************************/#include<reg51.h>#define uchar unsigned char#define GPIO_DIG P0#define GPIO_KEY P1uchar code DIG_CODE[17]={0x3f,0x06,0x5b,0x4f, 0x66,0x6d,0x7d,0x07, 0x7f,0x6f,0x77,0x7c, 0x39,0x5e,0x79,0x71};uchar KeyValue;void Delay10ms(); //延时10msvoid KeyDown(); //检测按键函数void main(void){ while(1) { KeyDown(); GPIO_DIG=~DIG_CODE[KeyValue]; } }void KeyDown(){ char a=0; GPIO_KEY=0x0f; //0000 1111 高四位全为0,低四位为1 if(GPIO_KEY!=0x0f) //读取按键是否按下 { Delay10ms(); //延时10ms进行消抖 if(GPIO_KEY!=0x0f) //再次检测键盘是否按下 { //测试列 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)) //检测按键松手检测 { Delay10ms(); //延时处理 a++; } } }}void Delay10ms() //误差 0us{ unsigned char a,b,c; for(c=1;c>0;c--) for(b=38;b>0;b--) for(a=130;a>0;a--);}
版权声明:本文为博主原创文章,未经博主允许不得转载。