標(biāo)題: 單片機(jī)如何檢測(cè)K1,K2按鍵同時(shí)按下,觸發(fā)LED反轉(zhuǎn),且不影響k1,與k2單獨(dú)工作 [打印本頁]

作者: 18796492606    時(shí)間: 2019-6-17 09:52
標(biāo)題: 單片機(jī)如何檢測(cè)K1,K2按鍵同時(shí)按下,觸發(fā)LED反轉(zhuǎn),且不影響k1,與k2單獨(dú)工作
      單片機(jī)如何檢測(cè)K1,K2同時(shí)按下,觸發(fā)LED反轉(zhuǎn),且不影響k1,與k2單獨(dú)工作

        if(k1==0)                  //檢測(cè)按鍵K1是否按下
        {        
                delay(1000);   //消除抖動(dòng) 一般大約10ms
                if((k1||k2)==0)
                {
                        led=~led;          //led狀態(tài)取反
                }
        }


                  if(k1==0)                  //檢測(cè)按鍵K1是否按下
                        {        
                                delay(1000);   //消除抖動(dòng) 一般大約10ms
                                if(k1==0)         //再次判斷按鍵是否按下
                                {
                                        led1=~led1;          //led1狀態(tài)取反
                                }
                                while(!k1);         //檢測(cè)按鍵是否松開
                        }
                        
                        if(k2==0)                  //檢測(cè)按鍵K1是否按下
                        {        
                                delay(1000);   //消除抖動(dòng) 一般大約10ms
                                if(k2==0)         //再次判斷按鍵是否按下
                                {
                                        led2=~led2;          //led2狀態(tài)取反
                                }
                                while(!k2);         //檢測(cè)按鍵是否松開
                        }



作者: 電子菜    時(shí)間: 2019-6-17 10:51
if(k1==0)                  //檢測(cè)按鍵K1是否按下
        {        
                delay(1000);   //消除抖動(dòng) 一般大約10ms
                if(k2==0)
                {
                        led=~led;          //led狀態(tài)取反
                }                 else    led1=~led1;
        }

if(k2==0)                  //檢測(cè)按鍵K2是否按下
                        {        
                                delay(1000);   //消除抖動(dòng) 一般大約10ms
                                if(k1==0)      
                                {
                                        led=~led;          //led2狀態(tài)取反
                                }

                               else  led2=~led2;  
                        }

作者: Shayven    時(shí)間: 2019-6-17 11:54
if(K1==0)  KEY=1;
if(K1==0)  KEY=2;
if(K1==0&&K2==0)  KEY=3;

消抖啥的自己加
作者: 18796492606    時(shí)間: 2019-6-17 14:52
Shayven 發(fā)表于 2019-6-17 11:54
if(K1==0)  KEY=1;
if(K1==0)  KEY=2;
if(K1==0&&K2==0)  KEY=3;

#include  "reg52.h"

typedef unsigned int u16;          //對(duì)數(shù)據(jù)類型進(jìn)行聲明定義
typedef unsigned char u8;

sbit k1=P3^1;         //定義P31口是k1
sbit k2=P3^0;         //定義P30口是k2
sbit led=P2^0;         //定義P20口是led
sbit led1=P2^1;         //定義P21口是led1
sbit led2=P2^2;         //定義P22口是led2

void delay(u16 i)
{
        while(i--);       
}

int key,linshizhi;

void keypros()
{
                 if(k1==0)                  //檢測(cè)按鍵K1是否按下
        {        
                delay(1000);   //消除抖動(dòng) 一般大約10ms
                if(k1==0)      
                {
                 key=1;         
                }  
        }

          if(k2==0)                  //檢測(cè)按鍵K2是否按下
        {        
                delay(1000);   //消除抖動(dòng) 一般大約10ms
                if(k2==0)      
                {
                   key=2;      
                }  
        }

                 if(k1==0&&k2==0)                  //檢測(cè)按鍵K1,K2是否按下
        {        
            delay(1000);   //消除抖動(dòng) 一般大約10ms
            if(k1==0&&k2==0)
            {
               key=3;      
            }      
        }
}

void xianshi()
{
   linshizhi=key;
   if(linshizhi==1){led=~led;}
   if(linshizhi==2){led1=~led1;}
   if(linshizhi==3){led2=~led2;}
   key=0;
}



void main()
{       
        while(1)
        {       
                keypros();  //按鍵處理函數(shù)
                xianshi();       
        }               
}
        實(shí)際實(shí)現(xiàn)不了



作者: 18796492606    時(shí)間: 2019-6-17 14:56
電子菜 發(fā)表于 2019-6-17 10:51
if(k1==0)                  //檢測(cè)按鍵K1是否按下
        {        
                delay(1000);    ...

void keypros()
{
         if(k1==0)                  //檢測(cè)按鍵K1是否按下
        {        
        delay(1000);   //消除抖動(dòng) 一般大約10ms
                if(k2==0)
                {
                     led=~led;          //led狀態(tài)取反
                }   else    led1=~led1;
        }

        if(k2==0)                  //檢測(cè)按鍵K2是否按下
        {        
        delay(1000);   //消除抖動(dòng) 一般大約10ms
                if(k1==0)      
                {
                led=~led;          //led2狀態(tài)取反
                }  else  led2=~led2;               
        }               
}

   現(xiàn)象不理想
作者: xianfajushi    時(shí)間: 2019-6-17 19:34
同時(shí)按下用“與邏輯”不是用“或邏輯”
作者: suncat0504    時(shí)間: 2019-6-17 22:46
樓主代碼中的第一個(gè)判斷中
if(k1==0)                  //檢測(cè)按鍵K1是否按下
        {        
                delay(1000);   //消除抖動(dòng) 一般大約10ms
                if((k1||k2)==0)
                {
                        led=~led;          //led狀態(tài)取反
                }
        }
改成
if(k1==0 && k2==0)                  //檢測(cè)按鍵K1和K2是否同時(shí)按下
        {        
                delay(1000);   //消除抖動(dòng) 一般大約10ms
                if(k1==0 && k2==0)  //檢測(cè)按鍵K1和K2是否同時(shí)按下
                {
                        led=~led;          //按鍵K1和K2同時(shí)按下, led狀態(tài)取反
                }
        }
即可
作者: 18796492606    時(shí)間: 2019-6-18 06:40
與不是“&&”這個(gè)嗎

作者: m182892    時(shí)間: 2019-6-18 08:13
你的程序K1 K2單獨(dú)按下也可以使LED取反,兩個(gè)同時(shí)按下也是,要用&&。
作者: egypt    時(shí)間: 2019-6-18 09:21
本帖最后由 egypt 于 2019-6-18 17:31 編輯

多兩個(gè)變量,記錄按住時(shí)間長(zhǎng)度,不會(huì)阻塞其它流程,這種邏輯最簡(jiǎn)單了

int  k1_count=0, k2_count=0;  

while(1)
{        // 注意 int 在 16bit 機(jī)器上可能會(huì)超過 32768 變成負(fù)數(shù)所以下文有30000這個(gè)常數(shù)
     if(k1==0) { if(k1_count<30000) k1_count++; } else k1_count=0;
     if(k2==0) { if(k2_count<30000) k2_count++; } else k2_count=0;
     
     // 所謂同時(shí)按下,代表 k1_count 和 k2_count 都位于 10ms-20ms 之間(10ms的去抖時(shí)間)
     // 假設(shè)一個(gè)先按了20ms 還沒松開,另外一個(gè)再按下,那就不叫同時(shí),叫先后按下
     // 去抖可以通過連續(xù)性判斷得到。k1_count 自加不滿 10次(每ms算一次)那就是接觸不好
     // 補(bǔ)充 k1_count k2_count 賦值是為了防止不停重入?梢愿鶕(jù)實(shí)際情況自行調(diào)整合理方案
     if( k1_count>=10 && k1_count<20 && k2_count>=10 && k2_count<20)
     {    led=~led;
          k1_count=30000; k2_count=30000;
     }
     ... delay(1ms); ...
}

作者: 電子菜    時(shí)間: 2019-6-18 09:21
18796492606 發(fā)表于 2019-6-17 14:56
void keypros()
{
         if(k1==0)                  //檢測(cè)按鍵K1是否按下

具體現(xiàn)象是什么?
作者: egypt    時(shí)間: 2019-6-18 10:28
我補(bǔ)充一下我回復(fù)的 10樓

*   關(guān)于去抖,回復(fù)中這種以連續(xù)性計(jì)時(shí)作為按下依據(jù),并不是最好(分時(shí)間片1ms能否準(zhǔn)時(shí)、以抖動(dòng)第一次作為計(jì)時(shí)依據(jù)之類的前置條件等),但是誤差也不大

*   關(guān)于人手按鍵,記憶中任天堂的游戲手柄有人可以按出17下每秒,一般人12-13下。兩次按鍵間隔差不多有 60ms 吧

*   所以如果覺得代碼中同時(shí)性條件太苛刻,可以把程序中 20 的數(shù)字改大,如 40 或 50 吧,以測(cè)試為準(zhǔn)
作者: quanquan12    時(shí)間: 2019-6-18 16:20
加多一條 if(看)
作者: quanquan12    時(shí)間: 2019-6-18 16:21
if(k1==0&&k2==0)  led=~led;  主體內(nèi)容就就是這樣,加多一條就可以實(shí)現(xiàn)了
作者: egypt    時(shí)間: 2019-6-18 18:12
quanquan12 發(fā)表于 2019-6-18 16:21
if(k1==0&&k2==0)  led=~led;  主體內(nèi)容就就是這樣,加多一條就可以實(shí)現(xiàn)了

關(guān)鍵是判斷 “同時(shí)按下瞬間”

如果 k1==0 和 k2==0 這樣判斷的話,按住 k1 和 k2 一個(gè)小時(shí)也會(huì)進(jìn) led 切換,按住一天也會(huì)進(jìn) led 切換的
作者: 尹子歸來    時(shí)間: 2019-6-18 18:56
18796492606 發(fā)表于 2019-6-17 14:52
#include  "reg52.h"

typedef unsigned int u16;          //對(duì)數(shù)據(jù)類型進(jìn)行聲明定義

這樣不會(huì)一直在反轉(zhuǎn)嗎,看上去也就一直是亮的,你可以在按鍵的時(shí)候加一步?jīng)]有按鍵按下的時(shí)候key為0,這樣顯示的時(shí)候只有檢測(cè)到key為0之后的下一步才判斷鍵值,不然不判斷就不會(huì)一直反轉(zhuǎn)看上去一直亮了

作者: suncat0504    時(shí)間: 2019-6-18 21:14
我的記憶中,消抖處理一般是20ms。
作者: lwh999995    時(shí)間: 2019-6-18 22:28
流程:
1. 判斷K1
2. 判斷K2
3. 判斷K1和K2
以上邏輯都不通,按K1,后面的K2是執(zhí)行不了的,當(dāng)你K1執(zhí)行成功了,那么兩個(gè)按鍵同時(shí)按下,那就邏輯錯(cuò)誤了,不可能跳過執(zhí)行,用兩個(gè)中斷,配合時(shí)間間隙和標(biāo)志位,可能行吧,很有可能還要用到GOTO
作者: sxywujizhidao    時(shí)間: 2019-6-18 23:47
查一下相關(guān)書籍也可以
作者: egypt    時(shí)間: 2019-6-19 08:37
suncat0504 發(fā)表于 2019-6-18 21:14
我的記憶中,消抖處理一般是20ms。

要看樓主的按鍵是什么樣的,便宜的軟膠墊碳膜,印象中差異比較大,鍋?zhàn)衅木秃眯?br />
在示波器量一下波形可以大致看出要去抖的時(shí)間去這兒看看別人是怎么測(cè)試的吧:
https://blog.csdn.net/xu8023636/article/details/79219880

作者: Shayven    時(shí)間: 2019-6-19 11:00
18796492606 發(fā)表于 2019-6-17 14:52
#include  "reg52.h"

typedef unsigned int u16;          //對(duì)數(shù)據(jù)類型進(jìn)行聲明定義

if(K1==0&&K2==0)  KEY=3;
else if(K1==0)  KEY=1;
else KEY=2;





歡迎光臨 (http://www.torrancerestoration.com/bbs/) Powered by Discuz! X3.1