我們有如圖一的一個矩陣鍵盤電路,首先我們知道當有一個按鍵按下去之后,相應的列線和行線之間就會短路(也可以說是直接相連),因此我們做掃描程序編碼時候,一般都是讓我們的8位輸出口首先輸出0xf0或者0x0f這樣子比較有特點的一些數值,因為一旦按鍵按下之后,就能和清晰的讀取回來,并且第一次判斷是哪一行,哪一列。然后在第二次把這些行全一或者列全一的數值進行對調,再讀入之后就能進行編碼了。
0.png (32.29 KB, 下載次數: 123)
下載附件
2017-9-27 23:12 上傳
圖1. 矩陣鍵盤電路
上述的鍵盤編碼方式是目前為止原理最簡單的方式,我也寫出了一個最簡化的程序,視頻里面也已經提到,這里不再贅述。
但是,我們仔細想一下以下的敘述方式。我第一次輸出0xf0,假設KEY2按下時,入讀的就應該是0xe0,那么假設我讓第二次輸出0x0e,試想下在讀取8這個電路的行列時候會變成什么樣子?是不是很容易得出0xde這個值?那這個0xde就是KEY2的編碼。比我之前講的方式少了好幾行代碼。因此我們在這里總結了行反轉法的基本流程:
(1) 端口輸出0xf0;
(2) 讀入,判斷是否為0xf0,如果不是表示有按鍵按下;
(3) 將讀入值的高4位和低4位交換,再輸出;
(4) 讀入端口的值,即按鍵的編碼。
0.png (686.2 KB, 下載次數: 130)
下載附件
2017-9-27 23:12 上傳
仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載)
0.png (24.92 KB, 下載次數: 138)
下載附件
2017-9-27 23:13 上傳
按鍵掃描程序流程.png (30.36 KB, 下載次數: 119)
下載附件
2017-9-27 19:34 上傳
單片機源程序如下:
- #include <reg52.h>
- typedef unsigned char u8;
- typedef unsigned int u16;
- sbit seg_sel = P1^4;
- sbit bit_sel = P1^5;
- #define DATA P0
- u8 code seg_tab[]={
- 0xc0,0xf9,0xa4,0xb0,
- 0x99,0x92,0x82,0xf8,
- 0x80,0x90,0x88,0x83,
- 0xc6,0xa1,0x86,0x8e
- };
- u8 code bit_tab[]={
- 0x10,0x20,0x40,0x80,
- 0x01,0x02,0x04,0x08
- };
- void delay(u16 num)
- {
- u16 x,y;
- for(x=num; x>0; x--)
- for(y=110; y>0; y--)
- {
- ;//延時1ms
- }
- }
- void display_led(u8 which_bit, u8 which_number)
- {
- bit_sel = 1;//Q[7..0]=D[7..0]
- DATA = bit_tab[which_bit];
- bit_sel = 0;//BIT[7..0]=0x80
- seg_sel = 1;//Q[7..0]=D[7..0]
- DATA = seg_tab[which_number];
- seg_sel = 0;
- delay(2);
- }
- u8 key_scan (void)
- {
- u8 temp,temp2;
- P2 = 0xf0;//讓P2輸出0xf0
- temp = P2;//讀入P2的值
- if (temp != 0xf0) //當P2讀入的值不等于0xf0
- {
- delay(5);//延遲5ms
- temp = P2;//再讀入P2口的值
- if (temp != 0xf0)//如果P2讀入的值不等于
- { //0xf0,就說明有按鍵按下
- temp2 = temp & 0xf0;//保留P2讀入值得高四位
- P2 = 0x0f;//再讓P2口輸出0x0f
- delay(1);
- temp = P2;//再讀入P2口的值
- temp2 |= temp;//保留此讀入值的低四位
- return temp2;
- }
- //while(P2!=0x0f);
- }
- }
- u8 encode(u8 cod)
- {
- u8 num;
- switch(cod)
- {
- case 0xee: num = 0;break;
- case 0xde: num = 1;break;
- case 0xbe: num = 2;break;
- case 0x7e: num = 3;break;
- case 0xed: num = 4;break;
- case 0xdd: num = 5;break;
- case 0xbd: num = 6;break;
- case 0x7d: num = 7;break;
- case 0xeb: num = 8;break;
- case 0xdb: num = 9;break;
- case 0xbb: num = 10;break;
- case 0x7b: num = 11;break;
- case 0xe7: num = 12;break;
- case 0xd7: num = 13;break;
- case 0xb7: num = 14;break;
- case 0x77: num = 15;break;
- default : break;
- }
- return num;
- }
- void main (void)
- {
- u8 cod,num;
- P0 = 0xff;
- P1 = 0xff;//端口初始化
- seg_sel = 0;//LE禁止鎖存
- bit_sel = 0;
- while (1)
- {
- cod = key_scan();//判斷以及編碼
- num = encode(cod);//解碼程序。得出數字
- display_led(1, num);//顯示程序
- }
- …………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼
0.png (52.49 KB, 下載次數: 149)
下載附件
2017-9-27 23:14 上傳
所有資料51hei提供下載:
矩陣鍵盤.rar
(1.17 MB, 下載次數: 105)
2017-9-27 19:26 上傳
點擊文件名下載附件
下載積分: 黑幣 -5
|