0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

矩陣鍵盤的按鍵識別方法_矩陣鍵盤掃描程序

姚小熊27 ? 來源:網(wǎng)絡整理 ? 作者:網(wǎng)絡整理 ? 2020-04-20 09:39 ? 次閱讀

矩陣鍵盤的按鍵識別方法

矩陣鍵盤的按鍵識別方法來自簡單日記網(wǎng)精選推薦。在學習有關矩陣鍵盤的時候,往往要學會矩陣鍵盤的按鍵識別方法,那么矩陣鍵盤的按鍵識別方法有哪些呢?小編帶著你來了解。

方法一:行掃描法

1、判斷鍵盤中有無鍵按下 將全部行線p1.4-p1.7置低電平,當然p1.0-p1.3為高電平(或許芯片內(nèi)部已經(jīng)將這些引腳它上拉),然后檢測列線的狀態(tài)。只要有一列的電平為低,則表示鍵盤中有鍵被按下,而且閉合的鍵位于低電平線與4根行線相交叉的4個按鍵之中。若所有列線均為高電平,則鍵盤中無鍵按下。

2、判斷閉合鍵所在的位置 在確認有鍵按下后,即可進入確定具體閉合鍵的過程。其方法是:依次將行線置為低電平,即在置某根行線為低電平時,其它線為高電平。在確定某根行線位置為低電平后,再逐行檢測各列線的電平狀態(tài)。若某列為低,則該列線與置為低電平的行線交叉處的按鍵就是閉合的按鍵。

方法二:先從p1口的高四位輸出低電平,低四位輸出高電平,從p1口的低四位讀取鍵盤狀態(tài)。再從p1口的低四位輸出低電平,高四位輸出高電平,從p1口的高四位讀取鍵盤狀態(tài)。將兩次讀取結果組合起來就可以得到當前按鍵的特征編碼。

矩陣鍵盤掃描程序

使用芯片STM8S003

所用端口:PD2~PD6, PA1~PA3

其中,PD3~PD6為輸出,PA1~PA3 / PD2為輸入(默認上拉)

/*

PortCom BIT7 BIT6 BIT5 BIT4 BIT3 BIT2 BIT1 BIT0

PD6 PD5 PD4 PD3 PD2 PA3 PA2 PA1

*/

程序如下:

/* 添加包含芯片的頭文件 */

#include《iostm8s103f3.h》

volatile unsigned char CF[4]; //按鍵觸發(fā)標志(表示4列,每一列同一行的

//值是一樣的但列標不一樣來區(qū)分不同列的鍵)

volatile unsigned char Cont[4];

unsigned char KeyVal; //鍵值

//unsigned char KeyOut[4] = {0xef,0xdf,0xbf,0x7f}; //4X4按輸出端控制

//unsigned char KeyOut[4] = {0x7f,0xbf,0xdf,0xef};

unsigned char KeyOut[4] = {0x3f,0x5f,0x6f,0x77}; //兩個端口組合4x4端口輸出控制

unsigned char PortCom; //兩個端口組合的端口

unsigned char cIn0,cIn1,cIn2,cIn3;

/*******************************************************************************

**函數(shù)名稱:void delay(unsigned int ms) Name: void delay(unsigned int ms)

**功能描述:大概延時

**入口參數(shù):unsigned int ms 輸入大概延時數(shù)值

**輸出:無

*******************************************************************************/

void delay(unsigned int ms)

{

unsigned int x , y;

for(x = ms; x 》 0; x--)

for(y = 1000 ; y 》 0 ; y--);

}

/*

**描述:新型4X4按鍵掃描程序 放在1ms-10ms中斷內(nèi)使用(十分穩(wěn)定不需要再寫消抖程序)

**備注:按鍵彈起時 keyVal = 0 單鍵按下 keyVal 有16個值,你自己程序可以針對不同值

**進行不同程序操作 keyVal單鍵值分別為

**0x01,0x02,0x04,0x08,

**0x11,0x12,0x14,0x18,

**0x21,0x22,0x24,0x28,

**0x31,0x32,0x34,0x38,

*/

void Key_Head()

{

unsigned char ReadData[4];

static unsigned char i;

/*

PortCom BIT7 BIT6 BIT5 BIT4 BIT3 BIT2 BIT1 BIT0

PD6 PD5 PD4 PD3 PD2 PA3 PA2 PA1

*/

cIn0 = 0;

cIn1 = 0;

cIn2 = 0;

cIn3 = 0;

if(++i》=4)i=0;

// PortCom = KeyOut[i]|0x0f; //忽略低4位

//輸出掃描

PD_ODR = KeyOut[i];

//輸入偵測

cIn0 = PA_IDR_bit.IDR1;

cIn1 = PA_IDR_bit.IDR2;

cIn2 = PA_IDR_bit.IDR3;

cIn3 = PD_IDR_bit.IDR2;

PortCom = (cIn3《《3) | (cIn2《《2) | (cIn1《《1) | cIn0;

ReadData[i] = (PortCom|0xf0)^0xff; //忽略高4位 取反

CF[i] = ReadData[i] & (ReadData[i] ^ Cont[i]);

Cont[i] = ReadData[i];

//輸出鍵值

switch(CF[i])//第i列

{

case 0x08: KeyVal = ((i《《4)+8);break;

case 0x04: KeyVal = ((i《《4)+4);break;

case 0x02: KeyVal = ((i《《4)+2);break;

case 0x01: KeyVal = ((i《《4)+1);break;

default:KeyVal = 0;break;

}

delay(30);

}

/*******************************************************************************

**函數(shù)名稱:void ALL_LED_Init() Name: void ALL_LED_Init()

**功能描述:初始化LED燈的IO口設為輸出

**入口參數(shù):無

**輸出:無

*******************************************************************************/

void ALL_LED_Init()

{

//LED1 Init

// PD_DDR_bit.DDR2 = 1; //設置端口PD-》2的輸入輸出方向寄存器為輸出方向

// PD_CR1_bit.C12 = 1; //設置PD2為推挽輸出

// PD_CR2_bit.C22 = 1; //設置PD2的輸出最大速度為10MHZ

//LED2 Init

PC_DDR_bit.DDR7 = 1; //設置端口PC-》7的輸入輸出方向寄存器為輸出方向

PC_CR1_bit.C17 = 1; //設置PC7為推挽輸出

PC_CR2_bit.C27 = 1; //設置PC7的輸出最大速度為10MHZ

//LED3 Init

PC_DDR_bit.DDR6 = 1; //設置端口PC-》6的輸入輸出方向寄存器為輸出方向

PC_CR1_bit.C16 = 1; //設置PC6為推挽輸出

PC_CR2_bit.C26 = 1; //設置PC6的輸出最大速度為10MHZ

//LED4 Init

PC_DDR_bit.DDR3 = 1; //設置端口PC-》3的輸入輸出方向寄存器為輸出方向

PC_CR1_bit.C13 = 1; //設置PC3為推挽輸出

PC_CR2_bit.C23 = 1; //設置PC3的輸出最大速度為10MHZ

}

/*******************************************************************************

**函數(shù)名稱:ALLKeyInit()

**功能描述:配置Key1 , Key2 , Key3輸入按鍵

**入口參數(shù):無

**輸出:無

*******************************************************************************/

void ALLKeyInit()

{

//PA1_Init

PA_DDR_bit.DDR1 = 0; //GPA-》PIN3 設置為輸入模式

PA_CR1_bit.C11 = 1; //GPA-》PIN3 帶上拉電阻輸入

PA_CR2_bit.C21 = 0; //GPA-》PIN3 禁止外部中斷

//PA2_Init

PA_DDR_bit.DDR2 = 0; //GPA-》PIN3 設置為輸入模式

PA_CR1_bit.C12 = 1; //GPA-》PIN3 帶上拉電阻輸入

PA_CR2_bit.C22 = 0; //GPA-》PIN3 禁止外部中斷

//PA3_Init

PA_DDR_bit.DDR3 = 0; //GPA-》PIN3 設置為輸入模式

PA_CR1_bit.C13 = 1; //GPA-》PIN3 帶上拉電阻輸入

PA_CR2_bit.C23 = 0; //GPA-》PIN3 禁止外部中斷

//PD2_Init

PD_DDR_bit.DDR2 = 0; //GPD-》PIN3 設置為輸入模式

PD_CR1_bit.C12 = 1; //GPD-》PIN3 帶上拉電阻輸入

PD_CR2_bit.C22 = 0; //GPD-》PIN3 禁止外部中斷

//PD3_Init

PD_DDR_bit.DDR3 = 1; //GPD-》PIN3 設置為輸入模式

PD_CR1_bit.C13 = 1; //GPD-》PIN3 帶上拉電阻輸入

PD_CR2_bit.C23 = 1; //GPD-》PIN3 禁止外部中斷

//PD4_Init

PD_DDR_bit.DDR4 = 1; //GPD-》PIN3 設置為輸入模式

PD_CR1_bit.C14 = 1; //GPD-》PIN3 帶上拉電阻輸入

PD_CR2_bit.C24 = 1; //GPD-》PIN3 禁止外部中斷

//PD5_Init

PD_DDR_bit.DDR5 = 1; //GPC-》PIN5 設置為輸入模式

PD_CR1_bit.C15 = 1; //GPC-》PIN5 帶上拉電阻輸入

PD_CR2_bit.C25 = 1; //GPC-》PIN5 禁止外部中斷

//PD6_Init

PD_DDR_bit.DDR6 = 1; //GPC-》PIN5 設置為輸入模式

PD_CR1_bit.C16 = 1; //GPC-》PIN5 帶上拉電阻輸入

PD_CR2_bit.C26 = 1; //GPC-》PIN5 禁止外部中斷

}

int main(void)

{

CLK_CKDIVR = 0x00; //內(nèi)部時鐘為1分頻

ALL_LED_Init(); //調(diào)用LED1初始化函數(shù)

ALLKeyInit(); //調(diào)用按鈕初始化函數(shù)

while(1)

{

Key_Head();

switch(KeyVal)

{

case 0x01:

{

PC_ODR ^= 0x80; //異或取反LED2使其亮滅

break;

}

case 0x02:

{

PC_ODR ^= 0x40; //異或取反LED3使其亮滅

break;

}

case 0x04:

{

PC_ODR ^= 0x08; //異或取反LED4使其亮滅

break;

}

case 0x08:

{

PC_ODR ^= 0x80; //異或取反LED2使其亮滅

PC_ODR ^= 0x40; //異或取反LED3使其亮滅

break;

}

case 0x11:

{

PC_ODR ^= 0x80; //異或取反LED2使其亮滅

break;

}

case 0x12:

{

PC_ODR ^= 0x40; //異或取反LED3使其亮滅

break;

}

case 0x14:

{

PC_ODR ^= 0x08; //異或取反LED4使其亮滅

break;

}

case 0x18:

{

PC_ODR ^= 0x80; //異或取反LED2使其亮滅

PC_ODR ^= 0x40; //異或取反LED3使其亮滅

break;

}

case 0x21:

{

PC_ODR ^= 0x80; //異或取反LED2使其亮滅

break;

}

case 0x22:

{

PC_ODR ^= 0x40; //異或取反LED3使其亮滅

break;

}

case 0x24:

{

PC_ODR ^= 0x08; //異或取反LED4使其亮滅

break;

}

case 0x28:

{

PC_ODR ^= 0x80; //異或取反LED2使其亮滅

PC_ODR ^= 0x40; //異或取反LED3使其亮滅

break;

}

case 0x31:

{

PC_ODR ^= 0x80; //異或取反LED2使其亮滅

break;

}

case 0x32:

{

PC_ODR ^= 0x40; //異或取反LED3使其亮滅

break;

}

case 0x34:

{

PC_ODR ^= 0x08; //異或取反LED4使其亮滅

break;

}

case 0x38:

{

PC_ODR ^= 0x80; //異或取反LED2使其亮滅

PC_ODR ^= 0x40; //異或取反LED3使其亮滅

break;

}

default:

{

KeyVal = 0;

break;

}

}

}

}

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 矩陣鍵盤
    +關注

    關注

    7

    文章

    205

    瀏覽量

    31384
收藏 人收藏

    評論

    相關推薦

    【xG24 Matter開發(fā)套件試用體驗】物聯(lián)網(wǎng)密碼柜之驅(qū)動矩陣鍵盤和OLED顯示器

    ;\"\"掃描矩陣鍵盤并返回按下的按鍵(row, col),如果按鍵已被按下則不重復返回\"\"\" # 逐行
    發(fā)表于 08-04 23:04

    基于51單片機矩陣鍵盤音樂電子琴電路圖proteus仿真及程序

    本資源內(nèi)容概要:? ? ? ?這是基于51單片機矩陣鍵盤音樂電子琴電路圖proteus仿真及程序設計包含了電路圖源文件(Altiumdesigner軟件打開)、C語言程序源代碼(kei
    發(fā)表于 06-21 14:32 ?1次下載

    鍵盤矩陣規(guī)格設計圖

    鍵盤矩陣規(guī)格設計圖
    發(fā)表于 06-19 14:29 ?0次下載

    用STM8L152使用一個矩陣鍵盤,信號一直處于低電平狀態(tài)的原因?

    想用STM8L152 使用一個矩陣鍵盤,根據(jù)的是行列掃描方法,但是代碼調(diào)試過程中只能檢測到列的信號(高低電平)的變化,信號一直處于低電平狀態(tài),這是怎么一回事?哪里出問題了?
    發(fā)表于 05-15 08:31

    采用NT33510顯示屏,如何通過矩陣鍵盤更改顯示屏上的數(shù)字內(nèi)容?

    我想實現(xiàn)一個這樣的功能:我的LCD當前顯示的是:新年快樂!2018! 想通過矩陣鍵盤將2018更改成2019,這個過程首先通過矩陣鍵盤上的左鍵移動到數(shù)字“8”上,然后在通過
    發(fā)表于 04-24 08:10

    stm32cubeMX如何配置4*4矩陣鍵盤?

    stm32cubeMX如何配置4*4矩陣鍵盤,求大佬給代碼指導一下,看了太多代碼,自己迷亂了
    發(fā)表于 03-28 09:27

    單片機如何讀取鍵盤數(shù)據(jù)

    等方面。 一、鍵盤的工作原理 鍵盤是一種常用的輸入設備,它主要通過按下/釋放按鍵的方式來向計算機發(fā)送相應的指令或數(shù)據(jù)。鍵盤通常由一個矩陣
    的頭像 發(fā)表于 01-04 17:09 ?1399次閱讀

    矩陣led掃描頻率怎么調(diào)

    矩陣LED掃描頻率是指LED矩陣在單位時間內(nèi)刷新顯示的次數(shù)。LED矩陣是由多個發(fā)光二極管(LED)組成的顯示設備,通過掃描控制來實現(xiàn)各個LE
    的頭像 發(fā)表于 01-02 17:30 ?1098次閱讀

    閱讀矩陣鍵盤的應用設計原理

    矩陣鍵盤以行和列的網(wǎng)格布線(盡管它們實際上不必以那個方向布置,例如在電子鋼琴鍵盤的例子中)。在微控制器上,行或列引腳被永久設置為輸入(在本文中,我們將堅持使用行連接實現(xiàn)此目的),而另一組引腳(在本文中,我們將堅持使用列連接實現(xiàn)此
    的頭像 發(fā)表于 11-28 11:09 ?699次閱讀
    閱讀<b class='flag-5'>矩陣</b><b class='flag-5'>鍵盤</b>的應用設計原理

    請問51單片機中如何從矩陣鍵盤中分解出獨立按鍵?

    請問51單片機中如何從矩陣鍵盤中分解出獨立按鍵?
    發(fā)表于 11-08 06:51

    基于單片機的獨立鍵盤矩陣鍵盤檢測

    獨立鍵盤,是指每一個鍵位單獨由一個IO口控制的按鍵。
    的頭像 發(fā)表于 10-31 15:00 ?1678次閱讀
    基于單片機的獨立<b class='flag-5'>鍵盤</b>與<b class='flag-5'>矩陣</b><b class='flag-5'>鍵盤</b>檢測

    基于單片機的矩陣鍵盤檢測案例

    /O端口資源,導致單片機與其它外部設備無法連接。在這種情況下,可以使用矩陣鍵盤,實現(xiàn)多個按鍵與單片機的連接,而且不浪費單片機的I/O端口資源。
    的頭像 發(fā)表于 10-31 14:35 ?1052次閱讀
    基于單片機的<b class='flag-5'>矩陣</b><b class='flag-5'>鍵盤</b>檢測案例

    請問怎么才能讓用矩陣鍵盤時蜂鳴器不叫(都在P1口)?

    我用的單片機學習板,矩陣鍵盤和蜂鳴器都在P1口上,使用矩陣時蜂鳴器一直叫,很無奈
    發(fā)表于 10-31 07:26

    關于51單片機的矩陣鍵盤掃描的兩種方法的困惑求解

    時間就去掃描按鍵,理論上和主程序的while(1)循環(huán)里代碼是否長和消耗時間應該無關,應該是很可靠的一種按鍵掃描
    發(fā)表于 10-26 08:02

    51單片機4*4矩陣鍵盤有什么好方法掃描識別?

    關于51單片機的4*4矩陣鍵盤,有什么你比較好掃描識別方法,麻煩告知一下,十分感謝!!
    發(fā)表于 10-24 06:23