找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1279|回复: 3
打印 上一主题 下一主题
收起左侧

关于单片机阵列按钮的检测问题

[复制链接]
跳转到指定楼层
楼主
ID:598105 发表于 2019-8-9 01:28 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
看到郭天祥新概念51单片机C语言教程的按钮阵列时,我完全被弄晕了,先上代码

我写的是用键盘阵列去改变P3管脚的电平,通过电平的改变去改变P2管脚从而驱动LED小灯,功能都实现了,但是我现在唯一搞不懂的地方就是检测矩阵键盘的地方
  1. 比如说我现在已经按下了矩阵键盘第2行的第2列,那么这个时候P3管脚的电平应该是11011101也就是0xDD
复制代码

哪位大神帮小弟解决一下这个问题啊,这个问题已经困扰我两天了


  1. #include <reg52.h>

  2. #define uchar unsigned char
  3. #define uint  unsigned int
  4.         
  5. uchar key;


  6. void delayms(uint xms) {
  7.         uint i, j;
  8.         for(i = xms; i > 0; i--)
  9.                 for(j = 110; j > 0; j--);
  10. }

  11. void matrixKeyScan() {
  12.         
  13.         uchar _temp;
  14.         
  15.         //检测第一行
  16.         P3    = 0xFE;
  17.         _temp = P3;
  18.         _temp = _temp & 0xF0;
  19.         if(_temp != 0xF0){
  20.                 delayms(10);
  21.                 _temp = P3;
  22.                 _temp = _temp & 0xF0;
  23.                 if(_temp != 0xF0){
  24.                         _temp = P3;
  25.                         switch(_temp) {
  26.                                 case 0xEE:
  27.                                         key = 0;
  28.                                         break;
  29.                                 case 0xDE:
  30.                                         key = 1;
  31.                                         break;
  32.                                 case 0xBE:
  33.                                         key = 2;
  34.                                         break;
  35.                                 case 0x7E:
  36.                                         key = 3;
  37.                                         break;
  38.                         }
  39.                         while(_temp != 0xF0){
  40.                                 _temp = P3;
  41.                                 _temp = _temp & 0xF0;
  42.                         }
  43.                 }
  44.         }
  45.         
  46.         //检测第二行
  47.         P3 = 0xFD;
  48.         _temp = P3;
  49.         _temp = _temp & 0xF0;
  50.         if(_temp != 0xF0){
  51.                 delayms(10);
  52.                 _temp = P3;
  53.                 _temp = _temp & 0xF0;
  54.                 if(_temp != 0xF0){
  55.                         _temp = P3;
  56.                         switch(_temp) {
  57.                                 case 0xED:
  58.                                         key = 4;
  59.                                         break;
  60.                                 case 0xDD:
  61.                                         key = 5;
  62.                                         break;
  63.                                 case 0xBD:
  64.                                         key = 6;
  65.                                         break;
  66.                                 case 0x7D:
  67.                                         key = 7;
  68.                                         break;
  69.                         }
  70.                         while(_temp != 0xF0){
  71.                                 _temp = P3;
  72.                                 _temp = _temp & 0xF0;
  73.                         }
  74.                 }
  75.         }
  76.         
  77.         //检测第三行
  78.         P3 = 0xFB;
  79.         _temp = P3;
  80.         _temp = _temp & 0xF0;
  81.         if(_temp != 0xF0){
  82.                 delayms(10);
  83.                 _temp = P3;
  84.                 _temp = _temp & 0xF0;
  85.                 if(_temp != 0xF0){
  86.                         _temp = P3;
  87.                         switch(_temp) {
  88.                                 case 0xEB:
  89.                                         key = 8;
  90.                                         break;
  91.                                 case 0xDB:
  92.                                         key = 9;
  93.                                         break;
  94.                                 case 0xBB:
  95.                                         key = 10;
  96.                                         break;
  97.                                 case 0x7B:
  98.                                         key = 11;
  99.                                         break;
  100.                         }
  101.                         while(_temp != 0xF0){
  102.                                 _temp = P3;
  103.                                 _temp = _temp & 0xF0;
  104.                         }
  105.                 }
  106.         }
  107.         
  108.         //检测第四行
  109.         P3 = 0xF7;
  110.         _temp = P3;
  111.         _temp = _temp & 0xF0;
  112.         if(_temp != 0xF0){
  113.                 delayms(10);
  114.                 _temp = P3;
  115.                 _temp = _temp & 0xF0;
  116.                 if(_temp != 0xF0){
  117.                         _temp = P3;
  118.                         switch(_temp) {
  119.                                 case 0xE7:
  120.                                         key = 12;
  121.                                         break;
  122.                                 case 0xD7:
  123.                                         key = 13;
  124.                                         break;
  125.                                 case 0xB7:
  126.                                         key = 14;
  127.                                         break;
  128.                                 case 0x77:
  129.                                         key = 15;
  130.                                         break;
  131.                         }
  132.                         while(_temp != 0xF0){
  133.                                 _temp = P3;
  134.                                 _temp = _temp & 0xF0;
  135.                         }
  136.                 }
  137.         }
  138.         
  139. }


  140. void main() {
  141.         
  142.         key = 99;
  143.         
  144.         while(1) {
  145.                 matrixKeyScan();
  146.                
  147.                 switch(key) {
  148.                         case 0:
  149.                                 P2 = 0xFE;
  150.                                 break;
  151.                         case 1:
  152.                                 P2 = 0xFD;
  153.                                 break;
  154.                         case 2:
  155.                                 P2 = 0xFB;
  156.                                 break;
  157.                         case 3:
  158.                                 P2 = 0xF7;
  159.                                 break;
  160.                         case 4:
  161.                                 P2 = 0xEF;
  162.                                 break;
  163.                         case 5:
  164.                                 P2 = 0xDF;
  165.                                 break;
  166.                         case 6:
  167.                                 P2 = 0xBF;
  168.                                 break;
  169.                         case 7:
  170.                                 P2 = 0x7F;
  171.                 }
  172.         }

  173. }
复制代码


分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:164602 发表于 2019-8-9 08:29 | 只看该作者
给你一个例子程序吧,比你的程序好、实用。
/*
本实例为矩阵键盘的实验
1、按不同的键,显示所按的键的数字。其实主要是实验出按的哪个键,程序判断对不对
   完全可以通过按键执行其它任务
2、矩阵键盘得到所按哪个键的过程相对复杂一点,注意理解。
*/

#include<reg51.h>

#define GPIO_DIG P0//数码管
#define GPIO_KEY P1//矩阵键盘

sbit LSA=P2^2;//数码管位码CBA分别为000第一位,001第二位,010第三位
sbit LSB=P2^3;//                   011第四位,100第五位,101第六位
sbit LSC=P2^4;//                                   110第七位,111第八位

unsigned char code DIG_CODE[17]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,
0x77,0x7c,0x39,0x5e,0x79,0x71};//0、1、2、3、4、5、6、7、8、9、A、b、C、d、E、F的显示码

unsigned char KeyValue;//用来存放读取到的键值

bit KeyState;        //记录按键的状态,0没有,1有

unsigned char DisplayData[8];//用来存放要显示的8位数的值,一个管分配一个数组

void Delay10ms();   //延时10ms
void KeyDown();                 //检测按键函数
void DigDisplay(); //动态显示函数

void main(void)
{
        KeyState=0;//初始化按键
        while(1)
        {
                KeyDown();//检测按键
                 if(KeyState==1)//有键按下才执行
                {
                        DisplayData[7]=DisplayData[6];//显示的内容左移一位
                        DisplayData[6]=DisplayData[5];
                        DisplayData[5]=DisplayData[4];
                        DisplayData[4]=DisplayData[3];
                        DisplayData[3]=DisplayData[2];
                        DisplayData[2]=DisplayData[1];
                        DisplayData[1]=DisplayData[0];
                        DisplayData[0]=DIG_CODE[KeyValue];
                       
                        KeyState=0;//显示完毕要初始化
                }
                DigDisplay();//显示
        }                               
}
/*******************************************************************************
* 函 数 名         : DigDisplay
* 函数功能                   : 使用数码管显示
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void DigDisplay()
{
        unsigned char i;
        unsigned int j;
        for(i=0;i<8;i++)//只有8个数码管
        {
                switch(i)         //位选,选择点亮的数码管,
                {
                        case(0):
                                LSA=0;LSB=0;LSC=0; break;//显示第0位
                        case(1):
                                LSA=1;LSB=0;LSC=0; break;//显示第1位
                        case(2):
                                LSA=0;LSB=1;LSC=0; break;//显示第2位
                        case(3):
                                LSA=1;LSB=1;LSC=0; break;//显示第3位
                        case(4):
                                LSA=0;LSB=0;LSC=1; break;//显示第4位
                        case(5):
                                LSA=1;LSB=0;LSC=1; break;//显示第5位
                        case(6):
                                LSA=0;LSB=1;LSC=1; break;//显示第6位
                        case(7):
                                LSA=1;LSB=1;LSC=1; break;//显示第7位       
                }
                GPIO_DIG=DisplayData[i];//发送段码
                j=10;                                                 //扫描间隔时间设定
                while(j--);       
                GPIO_DIG=0x00;//消隐
        }
}
/*******************************************************************************
* 函 数 名         : KeyDown
* 函数功能                   : 检测有按键按下并读取键值
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void KeyDown(void)//其实最简单的就是逐行扫描,就不难懂了。这里的行列扫描法有点
                                  //难理解,程序也复杂。
{
        unsigned char a=0;
        GPIO_KEY=0x0f;//测试列
        if(GPIO_KEY!=0x0f)
        {
                Delay10ms();//此处按键消抖,不可在测试列之后,那样显示时间不对,亮度很小
                if(GPIO_KEY!=0x0f)
                {
                        KeyState=1;//有按键按下
                        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;
                        Delay10ms();//此处延时为了消抖
                        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<200)&&(GPIO_KEY!=0xf0))//按键松手检测,a的值太小,就会在按下不松
                                                                                        //手时,产生多次按键的结果,现在有2s时间
                        {
                                Delay10ms();
                                a++;
                        }
                        a=0;
                }
        }
}


/*******************************************************************************
* 函 数 名         : Delay10ms
* 函数功能                   : 延时函数,延时10ms
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void Delay10ms(void)   //误差 0us
{
    unsigned char a,b,c;
    for(c=1;c>0;c--)
        for(b=38;b>0;b--)
            for(a=130;a>0;a--);
}
回复

使用道具 举报

板凳
ID:282095 发表于 2019-8-9 08:52 | 只看该作者
通过按键进行扫描矩阵的处理
回复

使用道具 举报

地板
ID:302534 发表于 2019-8-9 18:48 | 只看该作者
列扫描和竖扫描
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表