|
EDA365欢迎您登录!
您需要 登录 才可以下载或查看,没有帐号?注册
x
1 z, u1 { r! ?0 F
矩阵键盘是单片机编程中所使用的键盘." Y0 s2 Q, q& q& G& J! \
1.矩阵式键盘的结构与工作原理2 P! ?& ~. P8 K: j- U
在键盘中按键数量较多时,为了减少I/O口的占用,通常将按键排列成矩阵形式,如图1所示。在矩阵式键盘中,每条水平线和垂直线在交叉处不直接连通,而是通过一个按键加以连接。这样,一个端口(如P1口)就可以构成4*4=16个按键,比之直接将端口线用于键盘多出了一倍,而且线数越多,区别越明显,比如再多加一条线就可以构成20键的键盘,而直接用端口线则只能多出一键(9键)。由此可见,在需要的键数比较多时,采用矩阵法来做键盘是合理的。
1 f8 l5 n: u* F9 j6 [ 矩阵式结构的键盘显然比直接法要复杂一些,识别也要复杂一些,上图中,列线通过电阻接正电源,并将行线所接的单片机的I/O口作为输出端,而列线所接的I/O口则作为输入。这样,当按键没有按下时,所有的输出端都是高电平,代表无键按下。行线输出是低电平,一旦有键按下,则输入线就会被拉低,这样,通过读入输入线的状态就可得知是否有键按下了。具体的识别及编程方法如下所述。
3 l' i" z% X( C 2、矩阵式键盘的按键识别方法/ g3 @6 {# o1 U" N6 c& U- ~
确定矩阵式键盘上何键被按下介绍一种“行扫描法”。
# J. V% f1 S. e/ M' ]2 v0 ]0 r: x 行扫描法 行扫描法又称为逐行(或列)扫描查询法,是一种最常用的按键识别方法,如上图所示键盘,介绍过程如下。: Z F, V* L7 Y1 K# ?7 X
1、判断键盘中有无键按下 将全部行线Y0-Y3置低电平,然后检测列线的状态。只要有一列的电平为低,则表示键盘中有键被按下,而且闭合的键位于低电平线与4根行线相交叉的4个按键之中。若所有列线均为高电平,则键盘中无键按下。
+ R4 S+ K: G6 s& c0 R( Y 2、判断闭合键所在的位置 在确认有键按下后,即可进入确定具体闭合键的过程。其方法是:依次将行线置为低电平,即在置某根行线为低电平时,其它线为高电平。在确定某根行线位置为低电平后,再逐行检测各列线的电平状态。若某列为低,则该列线与置为低电平的行线交叉处的按键就是闭合的按键。
. G5 @: ^7 U3 j6 u2 p9 I 下面给出一个具体的例子:! I% z4 ]+ a$ {! w
图仍如上所示。8031单片机的P1口用作键盘I/O口,键盘的列线接到P1口的低4位,键盘的行线接到P1口的高4位。列线P1.0-P1.3分别接有4个上拉电阻到正电源+5V,并把列线P1.0-P1.3设置为输入线,行线P1.4-P.17设置为输出线。4根行线和4根列线形成16个相交点。3 |, n8 u V9 S" L- |% ?
1、检测当前是否有键被按下。检测的方法是P1.4-P1.7输出全“0”,读取P1.0-P1.3的状态,若P1.0-P1.3为全“1”,则无键闭合,否则有键闭合。
; B: o) b( d! } e8 p4 B 2、去除键抖动。当检测到有键按下后,延时一段时间再做下一步的检测判断。$ U5 N* F; Z$ f5 D: b2 q4 b, v
3、若有键被按下,应识别出是哪一个键闭合。方法是对键盘的行线进行扫描。P1.4-P1.7按下述4种组合依次输出:
6 @2 q. R7 N3 @% e4 ?8 ^% Q P1.7 1 1 1 0
, c% i6 @! L3 f7 T9 R7 C" Y P1.6 1 1 0 1
+ m% S/ D: }- z9 L7 P( m P1.5 1 0 1 1: g- _( ~( ^! Q- K1 K5 y
P1.4 0 1 1 1
; e* K$ u; Y3 Y 在每组行输出时读取P1.0-P1.3,若全为“1”,则表示为“0”这一行没有键闭合,否则有键闭合。由此得到闭合键的行值和列值,然后可采用计算法或查表法将闭合键的行值和列值转换成所定义的键值5 U7 i! x' L+ v5 z. l
4、为了保证键每闭合一次CPU仅作一次处理,必须却除键释放时的抖动。4 |7 f# ^7 ]* z0 M$ r
键盘扫描程序:: U* @$ i( ~( f
从以上分析得到键盘扫描程序的流程图如图2所示。程序如下
( ~! D3 |/ h% z8 `; d SCAN: MOV P1,#0FH
% A9 F' r, d) s5 T9 o4 ^$ b MOV A,P1
! c( F6 w: d2 J4 s- i ANL A,#0FH0 P" }% U, e- ?$ N. e/ L/ X
CJNE A,#0FH,NEXT1+ F, ?3 W, D; m; P
SJMP NEXT3* h# N, r+ R, w/ ?$ M
NEXT1: ACALL D20MS) l. t1 u( c5 k& ]2 \/ |" E& X% N. \& W
MOV A,#0EFH
# A2 k& G+ o: M9 K; O& R9 w# i NEXT2: MOV R1,A- v; q9 T1 n: i. W6 j
MOV P1,A
; [& y8 _ @/ r0 |# ]5 m: V) V MOV A,P1
' f$ m, `# B1 `& \2 ?1 Y1 b ANL A,#0FH
1 F( J/ A! J! u& S* M5 s. D) Q CJNE A,#0FH,KCODE;
) v' l6 ~# X1 u2 U8 R$ u: a2 t MOV A,R1
5 F! P4 z9 y. O; j SETB C
) ?# s2 P5 D) j) z RLC A
/ J) A+ q1 t! r: ~ JC NEXT2# L/ H( P z% h$ {2 i; H4 @
NEXT3: MOV R0,#00H
5 h) Y. O# B& r" X+ T RET! N' m8 T/ B+ p( o+ ]7 p5 f9 ]% u9 A
KCODE: MOV B,#0FBH/ ]% L- \' l" Z# D; C0 {
NEXT4: RRC A) w+ `, o. @2 @, e& E W3 W
INC B: m' c0 j) H% G8 v- W
JC NEXT4
! v' n( r/ g/ M# O$ p7 C6 v4 f MOV A,R1
$ h" J+ C/ G, Q" Z; R SWAP A: O0 [1 u" a" q" h. B
NEXT5: RRC A6 [5 q* x' G% e8 \% d6 T: M
INC B4 }8 ^- i. l) l& L) ^
INC B
0 [* G& w- a7 @ ~: K/ m( H1 K INC B
# q7 ^: E! p) a' i% G1 \& L INC B3 {+ }' A9 j. ^
JC NEXT5
7 B. R& _* g) X NEXT6: MOV A,P1
; g/ q8 |- b6 K2 G* E" G ANL A,#0FH
( R$ {* k6 R+ N0 B! } CJNE A,#0FH,NEXT68 T7 P& i3 V5 ]
MOV R0,#0FFH
% x2 a+ G8 N' l" G2 S! N* h# W# U RET5 j; Z3 y! y3 Z2 F' ]* x2 m
键盘处理程序就作这么一个简单的介绍,实际上,键盘、显示处理是很复杂的,它往往占到一个应用程序的大部份代码,可见其重要性,但说到,这种复杂并不来自于单片机的本身,而是来自于操作者的习惯等等问题,因此,在编写键盘处理程序之前,最好先把它从逻辑上理清,然后用适当的算法表示出来,最后再去写代码,这样,才能快速有效地写好代码。! F/ Q8 B/ M9 a. S( X7 ?
键盘是由一组规则排列的按键组成,一个按键实际上是一个开关元件,也就是说键盘是一组规则排列的开关。
9 Y' O$ Y' I6 B5 s# C 键盘工作原理
) t1 x9 G8 Y! |; a% x 1.按键的分类' P; [- h- j+ @" \
按键按照结构原理可分为两类,一类是触点式开关按键,如机械式开关、导电橡胶式开关等;另一类是无触点开关按键,如电气式按键,磁感应按键等。前者造价低,后者寿命长。目前,微机系统中最常见的是触点式开关按键。
3 I! |! J2 ~3 }1 d. k 按键按照接口原理可分为编码键盘与非编码键盘两类,这两类键盘的主要区别是识别键符及给出相应键码的方法。编码键盘主要是用硬件来实现对键的识别,非编码键盘主要是由软件来实现键盘的定义与识别。; @& V4 i! t5 _! @2 k
全编码键盘能够由硬件逻辑自动提供与键对应的编码,此外,一般还具有去抖动和多键、窜键保护电路,这种键盘使用方便,但需要较多的硬件,价格较贵,一般的单片机应用系统较少采用。非编码键盘只简单地提供行和列的矩阵,其它工作均由软件完成。由于其经济实用,较多地应用于单片机系统中。下面将重点介绍非编码键盘接口。
! i; ~. i, s0 s5 B4 _8 G 2.键输入原理
}- P9 `' J/ ^2 ~ 在单片机应用系统中,除了复位按键有专门的复位电路及专一的复位功能外,其它按键都是以开关状态来设置控制功能或输入数据。当所设置的功能键或数字键按下时,计算机应用系统应完成该按键所设定的功能,键信息输入是与软件结构密切相关的过程。# O0 {* A3 n8 Z- p$ z) z6 \3 L
对于一组键或一个键盘,总有一个接口电路与CPU相连。CPU可以采用查询或中断方式了解有无将键输入并检查是哪一个键按下,将该键号送入累加器ACC,然后通过跳转指令转入执行该键的功能程序,执行完后再返回主程序。+ ~# ^8 _5 D9 y
3.按键结构与特点
4 ^/ c9 A8 q$ r/ m& E/ E) |2 x 微机键盘通常使用机械触点式按键开关,其主要功能是把机械上的通断转换成为电气上的逻辑关系。也就是说,它能提供标准的TTL逻辑电平,以便与通用数字系统的逻辑电平相容。
) I! A+ ^; w/ J1 f. N, c( a 机械式按键再按下或释放时,由于机械弹性作用的影响,通常伴随有一定时间的触点机械抖动,然后其触点才稳定下来。其抖动过程如图7.2所示,抖动时间的长短与开关的机械特性有关,一般为5~10ms。
6 y; T1 Y8 U' _( A6 @ r& m4 \' ? 左图 按键触点的机械抖动) R0 H m, }. d1 K
在触点抖动期间检测按键的通与断状态,可能导致判断出错。即按键一次按下或释放被错误地认为是多次操作,这种情况是不允许出现的。为了克服按键触点机械抖动所致的检测误判,必须采取去抖动措施,可从硬件、软件两方面予以考虑。在键数较少时,可采用硬件去抖,而当键数较多时,采用软件去抖。
( @4 D, S9 J6 a" d 在硬件上可采用在键输出端加R-S触发器(双稳态触发器)或单稳态触发器构成去抖动电路,图7.3是一种由R-S触发器构成的去抖动电路,当触发器一旦翻转,触点抖动不会对其产生任何影响。
' Y" c- G# W6 X0 I' N2 m 电路工作过程如下:按键未按下时,a = 0,b = 1,输出Q = 1,按键按下时,因按键的机械弹性作用的影响,使按键产生抖动,当开关没有稳定到达b端时,因与非门2输出为0反馈到与非门1的输入端,封锁了与非门1,双稳态电路的状态不会改变,输出保持为1,输出Q不会产生抖动的波形。当开关稳定到达b端时,因a = 1,b = 0,使Q = 0,双稳态电路状态发生翻转。当释放按键时,在开关未稳定到达a端时,因Q = 0,封锁了与非门2,双稳态电路的状态不变,输出Q保持不变,消除了后沿的抖动波形。当开关稳定到达b端时,因a = 0,b = 0,使Q = 1,双稳态电路状态发生翻转,输出Q重新返回原状态。由此 右图 双稳态去抖电路0 g! h# U8 |7 W- y
可见,键盘输出经双稳态电路之后,输出已变为规范的矩形方波。5 x3 n- ]( c# z2 G& C7 n
软件上采取的措施是:在检测到有按键按下时,执行一个10ms左右(具体时间应视所使用的按键进行调整)的延时程序后,再确认该键电平是否仍保持闭合状态电平,若仍保持闭合状态电平,则确认该键处于闭合状态;同理,在检测到该键释放后,也应采用相同的步骤进行确认,从而可消除抖动的影响。
$ _1 r7 p% |/ H% E7 |* J 4.按键编码
+ ]5 P- o% k# k+ G+ Q- M 一组按键或键盘都要通过I/O口线查询按键的开关状态。根据键盘结构的不同,采用不同的编码。无论有无编码,以及采用什么编码,最后都要转换成为与累加器中数值相对应的键值,以实现按键功能程序的跳转。& N. e+ P- h/ V0 l9 o: n
5.编制键盘程序
/ |4 N) e. G/ f* E 一个完善的键盘控制程序应具备以下功能:) y8 I( K% W1 [6 j" Z' B! Z* q% W
(1)检测有无按键按下,并采取硬件或软件措施,消除键盘按键机械触点抖动的影响。6 J, R; I# X! ^
(2)有可靠的逻辑处理办法。每次只处理一个按键,其间对任何按键的操作对系统不产生影响,且无论一次按键时间有多长,系统仅执行一次按键功能程序。- Q, `& d+ W7 H
(3)准确输出按键值(或键号),以满足跳转指令要求。/ U1 _; w4 {6 Y: V$ r+ M& N/ \# Q
独立式按键& ~3 w) \% q4 B6 ^) m# |
单片机控制系统中,往往只需要几个功能键,此时,可采用独立式按键结构。$ w9 l3 ~+ T1 t8 ?
1.独立式按键结构7 |. }0 f) H% Y( n! {. b
独立式按键是直接用I/O口线构成的单个按键电路,其特点是每个按键单独占用一根I/O
6 f! E' W3 }9 B. E6 B# B 左图 独立式按键电路
4 ^. w' \' F5 F. w/ T# T- h 口线,每个按键的工作不会影响其它I/O口线的状态。独立式按键的典型应用如图7.4所示。
# T) X" G* }/ w/ o' e6 a 独立式按键电路配置灵活,软件结构简单,但每个按键必须占用一根I/O口线,因此,在按键较多时,I/O口线浪费较大,不宜采用。6 s& y& U/ g2 v! \, l
图7.4中按键输入均采用低电平有效,此外,上拉电阻保证了按键断开时,I/O口线有确定的高电平。当I/O口线内部有上拉电阻时,外电路可不接上拉电阻。4 p2 H y8 T( P9 V% w" ?
2.独立式按键的软件结构8 B! F8 h b8 n% R4 S5 |: E
独立式按键软件常采用查询式结构。先逐位查询每根I/O口线的输入状态,如某一根I/O口线输入为低电平,则可确认该I/O口线所对应的按键已按下,然后,再转向该键的功能处理程序。图7.4中的I/O口采用P1口,请读者自行编制相应的软件。
* ~+ q6 Y8 O9 n' M* T4 i0 L+ B0 Z: W 7.1.3 矩阵式按键1 M8 B! Q6 y- ]6 ], L5 K5 d
单片机系统中,若使按键较多时,通常采用矩阵式(也称行列式)键盘。
* G$ O6 R5 \- ~- U: P2 R$ I! _( W 1.矩阵式键盘的结构及原理; h* t# o) A: q5 z' `
矩阵式键盘由行线和列线组成,按键位于行、列线的交叉点上,其结构如下图所示。/ e, P6 Y' o9 z |% H
由图可知,一个4×4的行、列结构可以构成一个含有16个按键的键盘,显然,在按键数量较多时,矩阵式键盘较之独立式按键键盘要节省很多I/O口。
9 l( f* `2 w+ x$ v) F5 f 矩阵式键盘中,行、列线分别连接到按键开关的两端,行线通过上拉电阻接到+5V上。当无键按下时,行线处于高电平状态;当有键按下时,行、列线将导通,此时,行线电平将由与此行线相连的列线电平决定。这是识别按键是否按下的关键。然而,矩阵键盘中的行线、列线和多个键相连,各按键按下与否均影响该键所在行线和列线的电平,各按键间将相互影响,因此,必须将行线、列线信号配合起来作适当处理,才能确定闭合键的位置。+ T# |/ h2 W2 U @: k2 \
2.矩阵式键盘按键的识别
; N" y, V/ H9 c& @* f% \ 识别按键的方法很多,其中,最常见的方法是扫描法。下面以图7.5中8号键的识别为例来说明扫描法识别按键的过程。& ~( f% U4 X! k: p
按键按下时,与此键相连的行线与列线导通,行线在无键按下时处在高电平,显然,如果让所有的列线也处在高电平,那么,按键按下与否不会引起行线电平的变化,因此,必须使所有列线处在低电平,只有这样,当有键按下时,该键所在的行电平才会由高电平变为低电平。CPU根据行电平的变化,便能判定相应的行有键按下。8号键按下时,第2行一定为低电平,然而,第2行为低电平时,能否肯定是8号键按下呢?回答是否定的,因为9、10、11号键按下同样使第2行为低电平。为进一步确定具体键,不能使所有列线在同一时刻都处在低电平,可在某一时刻只让一条列线处于低电平,其余列线均处于高电平,另一时刻,让下一列处在低电平,依此循环,这种依次轮流每次选通一列的工作方式称为键盘扫描。采用键盘扫描后,再来观察8号键按下时的工作过程,当第0列处于低电平时,第2行处于低电平,而第1、2、3列处于低电平时,第2行却处在高电平,由此可判定按下的键应是第2行与第0列的交叉点,即8号键。1 X$ j8 X5 O7 F6 e+ F' o
3.键盘的编码- f& ~; _5 w5 j9 z4 x# L/ ?+ E
对于独立式按键键盘,因按键数量少,可根据实际需要灵活编码。对于矩阵式键盘,按键的位置由行号和列号唯一确定,因此可分别对行号和列号进行二进制编码,然后将两值合成一个字节,高4位是行号,低4位是列号。如图7.5中的8号键,它位于第2行,第0列,因此,其键盘编码应为20H。采用上述编码对于不同行的键离散性较大,不利于散转指令对按键进行处理。因此,可采用依次排列键号的方式对安排进行编码。以图7.5中的4×4键盘为例,可将键号编码为:01H、02H、03H…0EH、0FH、10H等16个键号。编码相互转换可通过计算或查表的方法实现。7 E+ Q! e/ y7 n9 C
4.键盘的工作方式
! _) c; l' V% ]7 P: v" Q) V* w8 h# Q5 L 在单片机应用系统中,键盘扫描只是CPU的工作内容之一。CPU对键盘的响应取决于键盘的工作方式,键盘的工作方式应根据实际应用系统中CPU的工作状况而定,其选取的原则是既要保证CPU能及时响应按键操作,又不要过多占用CPU的工作时间。通常,键盘的工作方式有三种,即编程扫描、定时扫描和中断扫描。8 z& X! V5 V# k) d; n$ ~2 n( ?8 a
1)编程扫描方式
9 M9 b- u! ^0 B 编程扫描方式是利用CPU完成其它工作的空余调用键盘扫描子程序来响应键盘输入的要求。在执行键功能程序时,CPU不再响应键输入要求,直到CPU重新扫描键盘为止。/ ^ @9 ~2 Y7 Q E7 w0 ~. e
键盘扫描程序一般应包括以下内容:, o( U3 S2 W& F
(1)判别有无键按下。/ w3 V8 N( Q- N% k* D4 P; b
(2)键盘扫描取得闭合键的行、列值。, d& n% f. O W D' B- I
(3)用计算法或查表法得到键值。
( G0 E2 z) L! D# v9 Y. n' y4 ` (4)判断闭合键是否释放,如没释放则继续等待。# x+ ~) d( i; F- G" m
(5)将闭合键键号保存,同时转去执行该闭合键的功能。9 D! x# O4 ~3 j9 j2 I0 Z
图7.6 8155扩展I/O口组成的矩阵键盘: x) d% R4 @! t: k' R
图7.6是一个4×8矩阵键盘电路,由图可知,其与单片机的接口采用8155扩展I/O芯片,键盘采用编程扫描方式工作,8155C口的低4位输入行扫描信号,A口输出8位列扫描信号,二者均为低电平有效。8155的IO/ 与P2.0相连, 与P2.1相连, 、 分别与单片机的 、 相连。由此可确定8155的口地址为:2 J' r. T/ [/ a$ o
命令/状态口:0100H(P2未用口线规定为0)4 T6 O! \4 ?7 L$ y6 ~& r9 H1 G
A口:0101H
, ]1 g9 b, @6 G* H/ s B口:0102H
5 o* z% f8 ~3 x4 m# ^" T C口:0103H
$ F3 S% S( I; A( A" C3 r/ ] 图7.6中,A口为基本输出口,C口为基本输入口,因此,方式命令控制字应设置为43H。在编程扫描方式下,键盘扫描子程序应完成如下几个功能:8 Z( X9 v e; J8 x U% S/ f; m
(1)判断有无键按下。其方法为:A口输出全为0,读C口状态,若PC0~PC3全为1,则说明无键按下;若不全为1,则说明有键按下。
/ m [- \& H- z$ V( d0 \; b; h (2)消除按键抖动的影响。其方法为:在判断有键按下后,用软件延时的方法延时10ms后,再判断键盘状态,如果仍为有键按下状态,则认为有一个按键按下,否则当作按键抖动来处理。
- Y3 _ L& k; } (3)求按键位置。根据前述键盘扫描法,进行逐列置0扫描。图7.6中,32个键的键值分布如下(键值由4位16进制数码组成,前两位是列的值,即A口数据,后两位是行的值,即C口数据,X为任意值):/ Q8 W' x$ u; j1 {9 i m2 w1 N) |
FEXE FDXE FBXE F7XE EFXE DFXE BFXE 7FXE
( X( a7 x2 u7 \- P4 }% u {, e FEXD FDXD FBXD F7XD EFXD DFXD BFXD 7FXD, I5 U, t2 c+ S @! ?
FEXB FDXB FBXB F7XB EFXB DFXB BFXB 7FXB: N) R& }' e7 o/ P) F f; X
FEX7 FDX7 FBX7 F7X7 EFX7 DFX7 BFX7 7FX7! P3 S% \! s3 p& ]
按键键值确定后,即可确定按键位置。相应的键号可根据下述公式进行计算:键号=行首键号+列号。图7.6中,每行的行首可给以固定的编号0(00H),8(08H),16(10H),24(18H),列号依列线顺序为0~7。" |# b% N2 k& \$ c; }7 y- t+ ^" K- ~
(4)判别闭合的键是否释放。按键闭合一次只能进行一次功能操作,因此,等按键释放后才能根据键号执行相应的功能键操作。
: U" l: M8 B9 e4 u$ y9 b/ M 键盘扫描程序流程图请参阅实训7图7.1中的主程序流程图。
7 B% w& t. V$ F, D& f7 S2 H3 _ 键盘扫描程序请参阅实训7源程序中的键盘查询程序、键盘扫描程序和按键查询子程序三部分。& \6 C/ t/ _7 u) M1 {
实训7是矩阵式键盘的一种典型应用,与图7.6相比,8155入口地址不同,矩阵键盘列数不同,再就是为兼顾键盘和显示,防抖所用的延时子程序由显示子程序替代。
4 b& n# c4 Q! Z0 t 2)定时扫描方式:8 ^1 e2 Q# u& q, N* y
定时扫描方式就是每隔一段时间对键盘扫描一次,它利用单片机内部的定时器产生一定时间(例如10ms)的定时,当定时时间到就产生定时器溢出中断,CPU响应中断后对键盘进行扫描,并在有键按下时识别出该键,再执行该键的功能程序。定时扫描方式的硬件电路与编程扫描方式相同,程序流程图如下图所示。5 G2 E) K ]: \
图中,标志1和标志2是在单片机内部RAM的位寻址区设置的两个标志位,标志1为去抖动标志位,标志2为识别完按键的标志位。初始化时将这两个标志位设置为0,执行中断服务程序时,首先判别有无键闭合,若无键闭合,将标志1和标志2置0后返回;若有键闭合,先检查标志1,当标志1为0时,说明还未进行去抖动处理,此时置位标志1,并中断返回。由于中断返回后要经过10ms后才会再次中断,相当于延时了10ms,因此,程序无须再延时。下次中断时,因标志1为1,CPU再检查标志2,如标志2为0说明还未进行按键的识别处理,这时,CPU先置位标志2,然后进行按键识别处理,再执行相应的按键功能子程序,最后,中断返回。如标志2已经为1,则说明此次按键已做过识别处理,只是还未释放按键,当按键释放后,在下一次中断服务程序中,标志1和标志2又重新置0,等待下一次按键。
9 q' e1 I6 i& v; d% r) ]2 Q 3)中断扫描方式
5 N, T4 X! w& s/ F0 N 采用上述两种键盘扫描方式时,无论是否按键,CPU都要定时扫描键盘,而单片机应用系统工作时,并非经常需要键盘输入,因此,CPU经常处于空扫描状态,为提高CPU工作效率,可采用中断扫描工作方式。其工作过程如下:当无键按下时,CPU处理自己的工作,当有键按下时,产生中断请求,CPU转去执行键盘扫描子程序,并识别键号。& V9 [+ r: a2 H4 i$ i: T2 q
左图是一种简易键盘接口电路,该键盘是由8051P1口的高、低字节构成的4×4键盘。键盘的列线与P1口的高4位相连,键盘的行线与P1口的低4位相连,因此,P1.4~P1.7是键输出线,P1.0~P1.3是扫描输入线。图中的4输入与门用于产生按键中断,其输入端与各列线相连,再通过上拉电阻接至+5V电源,输出端接至8051的外部中断输入端 。具体工作如下:当键盘无键按下时,与门各输入端均为高电平,保持输出端为高电平;当有键按下时,端为低电平,向CPU申请中断,若CPU开放外部中断,则会响应中断请求,转去执行键盘扫描子程序。3 V5 E, ~: o4 M4 U. n) t; L
这是初学时写的最简单、最详细、效率最高的矩阵键盘扫描程序,只用了四条常用命令(MOV/送数、JB/高电平转移、JMP/直接转移、RET/子程序返回),保证初学者一看就懂!本程序已经在本站电子实验板上验证通过,占用CPU时间少,效率高,被选作单片机的测试程序!
. Z% v3 L% ^, a' }5 L# a1 V; i s 矩阵按键扫描程序是一种节省IO口的方法,按键数目越多节省IO口就越可观,本程序的思路跟书上一样:先判断某一列(行)是否有按键按下,再判断该行(列)是那一只键按下。但是,在程序的写法上,站长采用了最简单的方法,使得程序效率最高。
3 _/ F/ T# w; U, N, `( ]' s+ b 本程序中,如果检测到某键按下了,就不再检测其它的按键,这完全能满足绝大多数需要,又能节省大量的CPU时间。另外,本人认为键盘用延时程序来消除抖动,完全是浪费时间。试想,如果不用中断执行(用中断执行需要更多的硬件资源)的方法来扫描键盘,每秒钟扫描20-100次,每次都要延时10-20MS的话,我们的单片机还有多少时间做正事呢?
4 g! r- m1 U, s9 F* E 其实,延时的这段时间,CPU可以做其它的事呀。所以,本键盘扫描程序的前面后面都可以加入少少代码,既可以达到完美的消抖动效果,又可以扩展其它的功能(例如按键封锁、按键长按等按键功能复用!)
+ ` M7 @; D4 X0 G+ ^ 本键盘扫描子程序名叫key,每次要扫描时用call key调用即可。以下子程序内容:
' [& m. P5 s$ [- [7 Z key:mov p0,#00001111b;上四位和下四位分别为行和列,所以送出高低电压检查有没有按键按下: h1 s8 Z; r' H1 v
jmp k10;跳到K10处开始扫描,这里可以改成其它条件转移指令来决定本次扫描是否要继续,例如减1为0转移或者位为1或0才转移,这主要用来增加功能,确认上一按键功能是否完成?是否相当于经过了延时?是否要封锁键盘?
) }' I A4 ^) `* l" ]* t8 I' ] goend:jmp kend;如果上面判断本次不执行键盘扫描程序,则立即转到程序尾部,不要浪费CPU的时间; ^3 ~( n/ c5 ~5 r" ?, n
k10:jb p0.0,k20;扫描正式开始,先检查列1四个键是否有键按下,如果没有,则跳到K20检查列28 h/ t r" [" o. K3 M7 v
k11:mov p0,#11101111b;列1有键按下时,P0.0变低,到底是那一个键按下?现在分别输出各行低电平$ c9 Y8 @1 F3 q" y) j7 r6 L
jb p0.0,k12;该行的键不按下时,p0.0为高电平,跳到到K12,检查其它的行
/ W0 R- O' Q1 ?9 X0 t0 f mov r1,#1;如果正好是这行的键按下,将寄存器R0写下1,表示1号键按下了
4 s" m) P* \ u$ P9 b k12:mov p0,#11011111b2 D; R8 z# ]3 i ~+ j/ V; N: u
jb p0.0,k13
% F( K+ Y# R# a mov r1,#2;如果正好是这行的键按下,将寄存器R0写下2,表示2号键按下了
8 Z: f3 t4 I+ C4 _8 [2 _* n k13:mov p0,#10111111b
1 \" I5 K& G" T* C jb p0.0,k148 `: ~# A5 I: }/ X5 b) Z6 |) [5 q4 ?
mov r1,#3;如果正好是这行的键按下,将寄存器R0写下3,表示3号键按下了
; @3 j. C& ^( r" G. K4 X+ Y3 ^ k14:mov p0,#01111111b# ?9 u3 b' |0 x' h& t! `
jb p0.0,kend;如果现在四个键都没有按下,可能按键松开或干扰,退出扫描(以后相同)
# N; P o. H& G9 y8 i mov r1,#4如果正好是这行的键按下,将寄存器R0写下4,表示4号键按下了% C& p: ~1 t3 k" z; X) @& U: L# g
jmp kend;已经找到按下的键,跳到结尾吧4 s& L1 o V& E7 S4 H# N' @- F
k20:jb p0.1,k30;列2检查为高电平再检查列3、4
7 w& e0 R7 N8 Z) k k21:mov p0,#11101111b;列2有健按下时,P0.0会变低,到底是那一行的键按下呢?分别输出行的低电平
/ Q( q: e8 }7 R4 W7 p! A6 E S6 L- m jb p0.1,k22;该行的键不按下时p0.0为高电平,跳到到K22,检查另外三行0 E( r- @& @; |) t' K
mov r1,#5;如果正好是这行的键按下,将寄存器R0写下5,表示5号键按下了(以后相同,不再重复了)
$ ^. B {* X E X, { k22:mov p0,#11011111b
; V: ^3 h% v5 l7 E. M+ F$ k0 ~ jb p0.1,k23& i7 u5 ^4 T' y, r' r+ C* q3 q
mov r1,#6' I$ J' v! {" U8 d7 a
k23:mov p0,#10111111b4 K6 v6 Q* g" X$ @+ }0 x$ ^
jb p0.1,k24' H! X+ t& z& e
mov r1,#7
1 K! b. L1 s" E" p1 x; c2 ? k24:mov p0,#01111111b
# H* ]2 C2 H( x y jb p0.1,kend: {5 g% l: w) B1 m
mov r1,#8
2 p5 h) C5 p/ f/ h7 V- O1 s+ j jmp kend;已经找到按下的键,跳到结尾吧(以后相同,不要重复了)( \: \) {" X8 c; R; D
k30:jb p0.2,k40# ^3 u- y0 F: }0 D9 b
k31:mov p0,#11101111b
" b1 X/ Z! L. \; T* s$ ] jb p0.2,k32) ?) b; G7 ~ L `3 c
mov r1,#9* v9 ?4 t. U* \5 u: M4 W5 d7 Y1 n
k32:mov p0,#11011111b+ D( w/ P5 T" [" f% b3 K
jb p0.2,k33
! J4 N2 E! K3 i1 X7 p mov r1,#10$ B. j2 R4 p7 N" c7 ~
k33:mov p0,#10111111b
' \0 m9 t* i4 p- W1 g$ U# q8 w5 ] jb p0.2,k34% B. W* W" U; G
mov r1,#11$ |- C5 o7 r1 O
k34:mov p0,#01111111b
& e! o a( x9 r" B, V* A jb p0.2,kend$ L/ T* G0 o+ y. u+ e5 y
mov r1,#12
. z4 ~* t2 Q' c; f jmp kend
9 \% V5 g, Y$ B" V6 {7 g+ _2 T k40:jb p0.3,kend: v4 L* j8 n/ N
k41:mov p0,#11101111b2 u% t2 J& |6 d# p4 l
jb p0.3,k42% M) C3 e. ~. ]1 U# \* W$ H
mov r1,#13; A* h& U9 r, i6 u4 l6 O% d
k42:mov p0,#11011111b, `! g! Q: g* i, S r* F. S9 [' n
jb p0.3,k43
7 x; T+ [" [! L& l mov r1,#14
% J+ Y% h% z& ~9 }. w* E7 l k43:mov p0,#10111111b, U& k. F8 X5 g& m6 `3 L0 S
jb p0.3,k44# I& b7 c" ?' l* w0 A
mov r1,#15
6 l9 {. E% }8 |$ i9 A k44:mov p0,#01111111b
% ~1 ]' n* Z; u" ^8 c) y jb p0.3,kend
/ k: Y$ u- Q/ h" Z mov r1,#166 j+ s4 d1 T9 a: u: {8 T
kend: ret
* }9 r( Z" I6 n* ^0 e& O 键盘扫描结束了,寄存器R1的值就直接表示了是那个键按下的,根据不同的键值去执行不同的程序,从而实现了十六个矩阵键盘扫描,同样原理,最多可以识别255个按键的矩阵扫描。' G) H2 L" F/ b* ?% a
我们可以每次键盘扫描开始时检查R0的值是否为0,只有在为0才扫描键盘,不为0就证明刚刚扫描过键值,相应的按键工作还没有完成。但是必须记得,每个按键命令执行完成后,要给R0写上0,表示可以扫描键盘。) z1 m5 ^1 [ C6 X H+ _, D
本键盘扫描程序的优点在于:不用专门的按键延时程序,提高了CPU效率,也不用中断来扫描键盘,节省了硬件资源。另外,本键盘扫描程序,每次扫描占用CPU时最短,不论有键按下或者无键按下都可以在很短的时间完成一次扫描。
3 t; ?5 I5 c' C, x+ O1 l |
|