找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2115|回復(fù): 6
收起左側(cè)

請大家?guī)兔Ψ治鲆幌逻@段鍵盤掃描單片機(jī)程序的執(zhí)行過程

[復(fù)制鏈接]
ID:476725 發(fā)表于 2019-5-21 19:42 | 顯示全部樓層 |閱讀模式
本帖最后由 Neighbor_John 于 2019-5-22 00:21 編輯

我學(xué)習(xí)到的鍵盤掃描是逐行或逐列掃描,而這段程序確不需要這么繁瑣。但是我感覺這段程序有很大的邏輯錯(cuò)誤,因?yàn)榧僭O(shè)按得第一個(gè)鍵,P1應(yīng)為0xee,然而※那一段代碼得出的P1為0xfe,因此這段程序不可能實(shí)現(xiàn),可我用仿真軟件實(shí)驗(yàn)后證實(shí)程序是可行的,而且這段程序也沒有檢測按鍵是否放開的部分,可仿真出來沒有任何問題,求大神看看這程序究竟怎么解釋。

單片機(jī)源程序如下:
  1. void key()         //獨(dú)立按鍵程序
  2. {
  3.         static uchar key_new = 0,key_l;
  4.         key_can = 20;                   //按鍵值還原
  5.         P1 = 0x0f;                                       
  6.         if((P1 & 0x0f) != 0x0f)      
  7.         {
  8.                 delay_1ms(1);                     //按鍵消抖動(dòng)
  9.                 if(((P1 & 0x0f) != 0x0f) && (key_new == 1))
  10.                 {                                                //確認(rèn)是按鍵按下
  11.                         key_new = 0;
  12.                         key_l= (P1 | 0xf0);
  13.                 ※      P1 = key_l ;  
  14.                         switch(P1)
  15.                         {
  16.                                 case 0xee:  key_can = 1;  break;  //得到按鍵值
  17.                                 case 0xde:  key_can = 4;  break;   //得到按鍵值
  18.                                 case 0xbe:  key_can = 7;  break;   //得到按鍵值
  19.                                 case 0x7e:  key_can = 10;  break;   //得到按鍵值        

  20.                                 case 0xed:  key_can = 2;  break;  //得到按鍵值
  21.                                 case 0xdd:  key_can = 5;  break;   //得到按鍵值
  22.                                 case 0xbd:  key_can = 8;  break;   //得到按鍵值
  23.                                 case 0x7d:  key_can = 0;  break;   //得到按鍵值

  24.                                 case 0xeb:  key_can = 3;  break;  //得到按鍵值
  25.                                 case 0xdb:  key_can = 6;  break;   //得到按鍵值
  26.                                 case 0xbb:  key_can = 9;  break;   //得到按鍵值
  27.                                 case 0x7b:  key_can = 11;  break;   //得到按鍵值
  28.       
  29.                                 case 0xe7:  key_can = 15;  break;  //得到按鍵值
  30.                                 case 0xd7:  key_can = 14;  break;  //得到按鍵值
  31.                                 case 0xb7:  key_can = 13;  break;   //得到按鍵值
  32.                                 case 0x77:  key_can = 12;  break;  //得到按鍵值  
  33.                         }                       
  34.                                 beep = 0;
  35.                                 delay_1ms(20);
  36.                                 beep = 1;
  37.                  }
  38.         }
  39.                 else
  40.                 {
  41.                         key_new = 1;
  42.                  }                       
  43.       
  44. }
復(fù)制代碼


回復(fù)

使用道具 舉報(bào)

ID:213173 發(fā)表于 2019-5-22 07:57 | 顯示全部樓層
void key()         //不是獨(dú)立按鍵程序,是4*4矩陣按鍵掃描程序
{
        static uchar key_new = 0,key_l;
        key_can = 20;                   //按鍵值還原
        P1 = 0x0f;  
        if((P1 & 0x0f) != 0x0f)
        {
                delay_1ms(1);                     //按鍵消抖動(dòng)
                //key_new是按鍵自鎖變量,防止按鍵重復(fù)響應(yīng),并做松手檢測
                if(((P1 & 0x0f) != 0x0f) && (key_new == 1))
                {                                 //確認(rèn)是按鍵按下
                        key_new = 0;
                        //假設(shè)按下第1鍵,此時(shí)讀取P1狀態(tài)是0000 1110
                        //P1|0xf0 邏輯運(yùn)算后保留低4位不變,高4位置1
                        key_l= (P1 | 0xf0);//將結(jié)果保存在臨時(shí)變量中1111 1110
                        P1 = key_l ;  //將1111 1110賦值P1
                        switch(P1)        //此時(shí)讀取P1值為1110 1110 即0xee,其它鍵以此類推
                        {//提示:端口寄存器由于接受外部控制信號,讀取的狀態(tài)不一定是賦值的狀態(tài)!
                                case 0xee:  key_can = 1;  break;  //得到按鍵值
                                case 0xde:  key_can = 4;  break;   //得到按鍵值
                                case 0xbe:  key_can = 7;  break;   //得到按鍵值
                                case 0x7e:  key_can = 10;  break;   //得到按鍵值
                                ......
回復(fù)

使用道具 舉報(bào)

ID:476725 發(fā)表于 2019-5-25 14:07 | 顯示全部樓層
wulin 發(fā)表于 2019-5-22 07:57
void key()         //不是獨(dú)立按鍵程序,是4*4矩陣按鍵掃描程序
{
        static uchar key_new = 0,key_l;

感謝解答。意思就是SWITCH語句判斷P1時(shí)不是根據(jù)P1的賦值是吧,那key_l的存在是不是沒什么意義,而且最后是怎么得到0xee的呢(如果按第一個(gè)鍵),我就是對這里很不明白。
回復(fù)

使用道具 舉報(bào)

ID:511890 發(fā)表于 2019-5-25 16:03 | 顯示全部樓層
Neighbor_John 發(fā)表于 2019-5-25 14:07
感謝解答。意思就是SWITCH語句判斷P1時(shí)不是根據(jù)P1的賦值是吧,那key_l的存在是不是沒什么意義,而且最后 ...

key_l是用來保持P1口的值的。按鍵一般是按下后就放開,也就是一瞬間的值。如果想要在按鍵松開后還要用到這個(gè)值的話,就要用一個(gè)變量來保存。
回復(fù)

使用道具 舉報(bào)

ID:155507 發(fā)表于 2019-5-26 10:04 | 顯示全部樓層
Neighbor_John 發(fā)表于 2019-5-25 14:07
感謝解答。意思就是SWITCH語句判斷P1時(shí)不是根據(jù)P1的賦值是吧,那key_l的存在是不是沒什么意義,而且最后 ...

4*4矩陣按鍵
       4  5  6  7
       |  |  |  |
3------+--+--+--+---
2------+--+--+--+---
1------+--+--+--+---
0------+--+--+--+---
       1  4  7  10 鍵

如果按第一個(gè)鍵, P1.0與P1.4連接,將1111 1110賦值P1,此時(shí)讀取P1值為1110 1110 即0xee,  因?yàn)镻1.0與P1.4已經(jīng)連接。
回復(fù)

使用道具 舉報(bào)

ID:476725 發(fā)表于 2019-5-29 15:15 | 顯示全部樓層
angmall 發(fā)表于 2019-5-26 10:04
4*4矩陣按鍵
       4  5  6  7
       |  |  |  |

明白了,謝謝
回復(fù)

使用道具 舉報(bào)

ID:476725 發(fā)表于 2019-5-29 15:16 | 顯示全部樓層
幻劍心 發(fā)表于 2019-5-25 16:03
key_l是用來保持P1口的值的。按鍵一般是按下后就放開,也就是一瞬間的值。如果想要在按鍵松開后還要用到 ...

明白了,謝謝
回復(fù)

使用道具 舉報(bào)

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表