新型的按键扫描程序能不能支持多按键啊

2020-01-19 19:35发布

在网上论坛经常看到这个新型的按键扫描程序,
新型的按键扫描程序

核心算法:
unsigned char Trg;

unsigned char Release;

unsigned char Cont;

void KeyRead( void )

{

    unsigned char ReadData = PINB^0xff;      // 1  读键值

Trg = ReadData & (ReadData ^ Cont);      // 2  得到按下触发值

Release=(ReadData^Trg^Cont);            //3   得到释放触发值

    Cont = ReadData;                       //4   得到所有未释放的键值

}
(1)       没有按键的时候 ReadData = 0x00;Trg = 0x00;Cont = 0x00;
(2)       第一次PB0按下的情况 ReadData = 0x01;Trg = 0x01;Cont = 0x01;
(3)       PB0按着不松(长按键)的情况 ReadData = 0x01;Trg = 0x00;Cont = 0x01;
(4)       按键松开的情况 ReadData = 0x00;Trg = 0x00;Cont = 0x00;

数码之家贴子在http://bbs.mydigit.cn/read.php?tid=584859,利用状态机思想扫描按键,感觉很有用,就想移植。但是移植过程中就发现问题了:理论是支持多个按键的扫描,但是我实验时怎么就只能实现一个按键的扫描呢?而且有点诡异现象!
我的实验部分代码:
#define        KEY1        =0x01
#define        KEY2        =0x02
#define        KEY3        =0x04
uchar Trg=0,Cont=0;
uchar num;

void keyscan(){                                
        unsigned char ReadData=(P2|0xf8)^0xff;//~(P2|0xf8);//(P2|0xf8)^0xff;//P2^0,1,2组需要处理,其他屏蔽
        Trg=ReadData &(ReadData^Cont);//判断是否点动,异或:不同就1
        Cont=ReadData;//判断是否长按
}
void tiem1(void) interrupt 3{
         keyscan();
        if(Trg&KEY1){               
                num++;
                if(num>15)
                        num=0;
        }
        if(Trg&KEY2){
                num--;
                if(num<1)
                        num=15;
        }
        if(Trg&KEY3){
                num=0;
        }
        display();
    TH1 = 0x63;              //20ms扫描一次
    TL1 = 0xc0;  
}
我的实验结果:
1、         void tiem1(void) interrupt 3{
         keyscan();
        if(Trg&KEY1){               
                num++;
                if(num>15)
                        num=0;
        }
        if(Trg&KEY2){
                num--;
                if(num<1)
                        num=15;
        }
        if(Trg&KEY3){
                num=0;
        }
        display();
    TH1 = 0x63;              //20ms扫描一次
    TL1 = 0xc0;  
}
则只有KEY1起作用,可以短按,长按。其他按键无效。
2、         keyscan();
        if(Trg&KEY2){
        }

        if(Trg&KEY3){
        }
        if(Trg&KEY1){               
        }
则只有KEY2起作用,其他按键无效。
3、keyscan();
        if(Trg&KEY3){
        }

        if(Trg&KEY1){               
        }
        if(Trg&KEY2){
        }
则只有KEY3起作用,其他按键无效。
结论:谁在前,谁起作用,不能多按键扫描。

这就是上面我所说的诡异了。搞了很久都没搞明白我实在是菜啊

请各位亲指点下迷津

友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
20条回答
Playboy_xixi
1楼-- · 2020-01-20 16:53
 精彩回答 2  元偷偷看……
Playboy_xixi
2楼-- · 2020-01-20 22:06
ft_cs 发表于 2013-6-23 00:13
用这个程序试试,保证好用,20Ms扫描一次KeyScan()
//按键检测
unsigned char KeyPressDown=0x00;                                  // ...

P2=0x0f;                                                                        //高四位输出低电平
        KEY_H=P2&0x0f;                                                                //读低四位
        P2=0xf0;                                                                        //低四位输出低电平
        KEY_L=P2&0xf0;                                                                //读高四位
        CurrReadKey=~(KEY_H | KEY_L);                                 //取反
这样折腾,是不是4*4按键啊?
其后的按键出来,真的很复杂。不过,貌似思路跟文中提到的差不多
CurrKey=ReadData
KeyPressDown=Trg
KeyRelease=Release
LastKey=Cont
不过,当我看到switch (KeyPressDown)就恍然大悟,这不是switch(Trg)吗?
既然我的诡异现象是:谁在前,谁起作用
原因其实就是要让三个按键的触发享有相同的优先权,显然,用if是想不通
前些天想过用switch结构,但是自己思维真的拘束的在Trg&KEY上,三个按键没有相同的表达式。
其实,Trg本身就是很好的switch表达式
void tiem1(void) interrupt 3{
        keyscan();
        switch(Trg)
        {        case 0x01:
                        break;
                case 0x02:
                        break;
                case 0x04:
                        break;
                default :break;
        }
        display();
    TH1 = 0x63;              
    TL1 = 0xc0;  
}
问题解决。
弱弱回复说多一句,按操作中的优先级:&>^>|;
感谢启发
Bicycle
3楼-- · 2020-01-21 03:56
试下这个
unsigned char ReadData=(P2^0xff)&0x07;//P2^0,1,2组需要处理,其他屏蔽
Playboy_xixi
4楼-- · 2020-01-21 04:03
Bicycle 发表于 2013-6-23 16:12
试下这个
unsigned char ReadData=(P2^0xff)&0x07;//P2^0,1,2组需要处理,其他屏蔽

问题已经解决,就在上楼。
这个已经在注释里了
unsigned char ReadData=(P2|0xf8)^0xff;//~(P2|0xf8);//(P2|0xf8)^0xff;//P2^0,1,2组需要处理,其他屏蔽
redchina
5楼-- · 2020-01-21 05:54
好,MARK,后面也试用下这个程序
barryliu
6楼-- · 2020-01-21 10:50
 精彩回答 2  元偷偷看……

一周热门 更多>