按键输入实验/ p5 g' M# L3 p. l; [6 ], N
本章,我们向大家介绍IO口作为输入使用的操作方法,我们将利用板载的3个按键来控制LED灯亮和灭以及和蜂鸣器的开和关。
! Z% ~. t y* C! s+ E# |$ ^
! M4 i, G( W/ |* X- l12.1 按键输入简介; M; L0 H5 [" ]
几乎每个开发板都会板载有独立按键,因为按键用处很多。常态下,独立按键是断开的,按下的时候才闭合。每个独立按键会单独占用一个IO口,通过查询IO口的高低电平来判断按键的状态。按键在闭合和断开的时候,都存在抖动现象,即按键在闭合时不会马上就稳定的连接,断开时也不会马上断开,这是机械触点,无法避免。独立按键抖动波形图如下:% a: x M5 {/ {9 w
! J: a' {8 R) f8 k5 ]1 r$ R- Z+ f
5 R6 b9 {& s Y
# z5 U& h& T+ ~: Z9 I图12.1. 1独立按键抖动波形图# u$ t4 U( W9 G' Q0 T2 v
图中的按下抖动和释放抖动的时间大概都为10ms。为了避免抖动可能带来的误操作,我们要做的措施就是给按键消抖。消抖方法分为硬件消抖和软件消抖,我们常用软件的方法消抖。0 {5 T, ^ N) M
软件消抖:检测到按键按下后,一般进行10ms延时,用于跳过抖动的时间段,如果消抖效果不好可以调整这个10ms延时,因为不同类型的按键抖动时间可能有偏差。按键按下以后,待延时过后再检测按键状态,如果测试到按键没有按下,那我们就判断这是抖动或者干扰造成的;如果测试到还是按下的状态,那么我们就认为这是按键真的按下了。对按键释放的判断同理。
% w6 y8 K. E, Q6 p" `* n& A硬件消抖:利用R-S触发器或者单稳态触发器构成消抖电路,或者利用RC积分电路吸收振荡脉冲的特点来达到消抖的效果。
7 c V7 c/ W6 `$ @- R12.2 硬件设计8 E2 h: Z5 @3 z
1.例程功能1 j9 V$ K: I# H. Z# t/ g( }
通过开发板上的三个独立按键控制LED灯:WKUP控制LED0翻转,KEY1控制LED1翻转,KEY0控制蜂鸣器翻转。3 o% ~$ w) F6 j% _# |( x
2.硬件资源
4 W0 @/ D- e! z+ ~! }: [1 A. O- |0 U( D+ ?
2 f! f9 m- {2 Q表12.2. 1硬件资源
4 {& i- o" ?- [( {" |: D3 K8 K; J4 {3. 原理图
$ f7 s T6 a1 ?8 B! R1 d, d- YLED和BEEP的原理图我们前面已经涉及,独立按键硬件部分的原理图如下图所示:
7 t% Z/ z$ B6 ^% t9 S% e# t4 d0 H7 G F* z0 r
3 w, P, T# d1 {# o' f# ?. X- K
; s( H O3 K" P: Q1 k5 ?8 [
图12.2. 1独立按键与STM32MP157连接原理图3 {0 f. o- L" y: X, ^/ ?
这里需要注意的是:KEY0和KEY1是低电平有效的,而WK_UP是高电平有效的,并且KEY0和KEY2接了一个10k的上拉电阻,而WK_UP外部没有下拉电阻,所以WK_UP的配置中,一定要通过软件来开启STM32MP157内部下拉。
! O/ Q1 m5 l" W1 X3 k, }' E/ }8 `9 {# v
12.3 软件设计
/ U4 e' W$ N/ E2 [; u' L本实验配置好的实验工程已经放到了开发板光盘中,路径为:开发板光盘A-基础资料\1、程序源码\11、M4 CubeIDE裸机驱动例程\CubeIDE_project\ 5 KEY。
" e u. W; r6 c+ c& p: d12.3.1 程序设计流程
' Q% z* }' t3 @( r2 r2 h本章节我们通过HAL库的HAL_GPIO_ReadPin函数(实际上是操作IDR寄存器)来读取按键对应GPIO引脚电平状态,从而判断按键是否有按下。实验程序的设计流程如下:
8 @2 \; Z* R& n0 t# {0 h% A+ NKEY0按下,LED0翻转;KEY1按下,LED1翻转,WKUP按下,蜂鸣器翻转。
: c1 R* e6 }* Q6 z4 ^, {, F! X' |
) `5 X7 h. V* t( Y* p# H$ {
9 T! w6 B2 R% I5 g5 T7 \图12.3.1. 1程序设计流程图7 @) U1 y; t# l
: o3 [% s9 i- A6 }3 P
12.3.2 GPIO功能引脚配置8 a" T; j; x7 ^
新建一个工程KEY,或者可以直接在上一章节的BEEP工程的BEEP.ioc文件中配置。按照前面的配置方法配置好LED0、LED1、BEEP和按键,要注意以下3点:* p- I6 N, _/ x' V) y( a3 k
①这里,除了按键WKUP配置为下拉以外,其它引脚均配置为上拉了;+ D: t! S; `) W# K2 U9 g/ c3 L
②User Lable名字要和前面的实验的一致,因为我们后面会用到前面实验的代码;
! ]) h- |. F" ~③记得要配置引脚给M4内核使用,这点我们在前面的实验中都有提到,后面的实验就不再赘述。
; k& S8 m5 }, L* m; c# `3 C引脚配置结果如下图所示:( \ P: k- v; E6 k# l5 k
+ ]. M; h' ]9 h- S1 x
7 w) s& @: k; P. \# W) O
/ I1 K+ X$ Z9 O6 W图12.3.2. 1 GPIO功能引脚配置
' F2 \4 o8 x W. ^
- z! H, i" [ z; H12.3.3 时钟和工程配置. j% G O# f" F4 P, b
我们采用默认内部高速时钟HSI(64MHz)。同时在Project Manager窗口勾选此项,配置独立生成对应外设的初始化.h和.c 文件:3 l4 C& {) C0 F6 d/ T, W3 {
, W: p% @; Y8 }" c
3 s% y# d1 m% Z9 ]7 M
! U3 n* j: P' Q7 H2 U/ \2 v/ X6 a
图12.3.2. 2配置生成外设独立的初始化文件
0 G9 D5 m, J( P5 P+ l% S5 ^; x( R6 f/ U; @2 l# z2 u
12.3.4 生成工程/ d' i9 V$ q2 y" c. P0 @
配置好之后,按下键盘的“Ctrl+S”组合键保存保存 KEY.ioc 文件,系统开始生成初始化代码。
- s1 d$ l2 e ^ F" v6 ?) q4 Y% X/ U! \& E- U
6 h6 r5 ~4 a0 G; x! v* L
' R+ k. Y/ i* T5 n图12.3.4. 1生成KEY工程
; s3 w0 v$ ]; l9 C" N/ A
5 o# j+ g6 w& P0 X12.3.5 控制逻辑代码实现$ B% B2 I0 z, E& y
1.新建用户文件8 G" J. }/ O/ ?& g4 |8 I/ B
将上一章BEEP工程的BSP文件夹直接复制粘贴到本工程的CM4工程的Core文件夹下(这次直接拷贝到Core文件夹下,就不需要再设置源文件和工程关联了)。然后在BSP的Include目录下新建key.h头文件,在BSP的根目录下新建key.c源文件,如下图:
5 l- B" E; [6 ~4 u' v6 r9 N8 T( W: p0 E7 ^) X+ v
) e% ^$ }8 F6 P4 h; r3 t2 C, `& F h' ~
图12.3.5. 1新建文件
6 P2 H1 T4 l4 I+ k; U/ A2. 添加用户驱动代码, [4 g* \2 v& q! O
LED和蜂鸣器的代码不变,我们在key.h、key.c和main.c文件中添加如下代码。! C3 [4 g3 S% M7 \1 {
key.h文件代码如下:
8 Z: l. U# v; T) i% ^key.h文件代码
" V3 V3 n4 s) v6 k; b% m$ l2 Z
; A" ? R3 `6 H/ _* f- 1 #ifndef __KEY_H
) r$ b o- h2 { - 2 #define __KEY_H3 X& X) y2 g0 }( h2 }5 s1 Y' k; a
- 3
; t" p! _4 I5 W2 E/ E6 j/ D. N @ - 4 #include"gpio.h"- p4 ^3 S; f4 {% d1 Z2 _2 u
- 5 /* 读取KEY0引脚 */
3 _* _' K" C' E5 ~ - 6 #define KEY0 HAL_GPIO_ReadPin(KEY0_GPIO_Port, KEY0_Pin)4 F, M$ L( y1 N: U
- 7 /* 读取KEY1引脚 */
' d% l0 T6 D4 H* N5 Z - 8 #define KEY1 HAL_GPIO_ReadPin(KEY1_GPIO_Port, KEY1_Pin)
" C: _- a+ _( ^# R) J - 9 /* 读取WKUP引脚 */
. m; X* Q9 y; n - 10 #define WKUP HAL_GPIO_ReadPin(WKUP_GPIO_Port, WKUP_Pin)8 h6 L! H7 x& ^( M L
- 11
s o/ t, B& _ k, u; o" {! t. P! s- ^% V - 12 #define KEY0_PRES 1 /* KEY0按下 */
" c3 P5 `$ }" F$ z+ A- x5 h" B - 13 #define KEY1_PRES 2 /* KEY1按下 */
0 N2 Z: y1 f! @3 ~/ k y5 j - 14 #define WKUP_PRES 3 /* WKUP按下 */: q5 g0 G. L! ^/ n
- 15
# c3 M$ ^/ X3 R/ y - 16 uint8_t key_scan(uint8_t mode); /* 按键扫描函数 */
6 z" H0 ^! I' Q5 }; q/ E3 h - 179 a& S! E; Q: b2 W1 } [
- 18 #endif
复制代码
" q8 S3 o$ c$ f9 n, l" Hkey.h头文件很简单:/ y; O1 L* \+ _ V
第6~10行,通过HAL_GPIO_ReadPin函数读取GPIO的电平值,也就是读取按键电平值。* s8 K( b1 X# C+ { o
第12~14行,定义3个宏,分别对应3个按键按下。1 A$ d5 ^/ z+ [' q0 b! R- t/ y
第16行,声明按键扫描函数key_scan。
, X9 d! h( e' S8 A9 [key.c文件代码如下:, t2 }- n+ M8 _$ G5 z. j
5 z; c/ G: H1 F/ n- u
key.c文件代码
) G9 h0 L8 L- {; g W8 S6 x
; l5 Q! ?* R' E0 Q4 i% A8 h- 1 #include "./Include/key.h"
# C% V5 e" p; Y3 _; ~ - 2 #include "./Include/led.h"
4 {# y9 b3 }" ]5 o. Y& \9 J - 3 #include "./Include/beep.h"
( p8 o* z8 @3 V [: g6 F) { - 4 ! t5 K: R9 s: \" H" L- z
- 5 /**9 [& s' j( h+ N6 q
- 6 * @brief 按键扫描函数
1 t3 N2 O$ \9 u - 7 * @note 该函数有响应优先级(同时按下多个按键): WKUP > KEY1 > KEY0!0 R) e: Y0 G( G2 Q ~+ F$ r" m# |
- 8 * @param mode:0 / 1, 具体含义如下:- ]" m! ~4 v9 U* t& m+ ~& U F0 _
- 9 * @arg 0, 不支持连续按(当按键按下不放时, 只有第一次调用会返回键值,0 S. Y) k+ S: P7 O9 t1 y* H% V! O
- 10 * 必须松开以后, 再次按下才会返回其他键值)
; {3 z$ }4 Z9 M* h' m2 {' y - 11 * @arg 1, 支持连续按(当按键按下不放时, 每次调用该函数都会返回键值)( X D9 b, {1 \* g, V" g
- 12 * @retval 键值, 定义如下:
) K$ O1 E8 ~; m/ P5 f9 n - 13 * KEY0_PRES, 1, KEY0按下# Q: u7 h9 h) s' I8 _2 O
- 14 * KEY1_PRES, 2, KEY1按下1 L g1 e' U* s) A, |) D" H, m
- 15 * WKUP_PRES, 3, WKUP按下
9 t) h0 W% W" t% Q7 O - 16 */$ B& ]1 [; u1 K: K2 U w
- 17 uint8_t key_scan(uint8_t mode)
; i5 Y A4 C9 Q! R+ u: i - 18 {: t& C7 x, W$ ~7 P5 a; B1 v
- 19 static uint8_t key_up = 1; /* 按键按松开标志 */# j# L. u& W Q; @
- 20 uint8_t keyval = 0;
u5 @7 ?2 ~! ~0 m - 21" \! F% a) g e
- 22 if (mode) key_up = 1; /* 支持连按 */. I! H$ G% p! O4 F
- 23 /* 按键松开标志为1, 且有任意一个按键按下了 */; i: t* e2 m6 n
- 24 if (key_up && (KEY0 == 0 || KEY1 == 0 || WKUP == 1)) ) [# {3 ^/ Z a, D) O
- 25 {2 H# `( u1 c. @# W! @
- 26 HAL_Delay(10); /* 去抖动,后面会换成高精度延时函数! */# ]8 s2 t8 h$ U' |
- 27 key_up = 0;
( Z6 K5 x% T- P6 [$ n# j8 | - 28; `; E, c K m
- 29 if (KEY0 == 0) keyval = KEY0_PRES;
+ e# f1 U" O' H- p8 f; X& U8 b - 30
, i l# w6 v! x" N% f7 W! a - 31 if (KEY1 == 0) keyval = KEY1_PRES;
' N4 a* D" s% x, k - 32
- n0 p4 B5 k: O1 {, C2 K5 [ - 33 if (WKUP == 1) keyval = WKUP_PRES;
/ m; l3 D% _1 x - 34 }/ i4 @& {, m- w! P+ j* l1 V
- 35 /* 没有任何按键按下, 标记按键松开 */) M$ O; ~. D# W1 W, V
- 36 else if (KEY0 == 1 && KEY1 == 1 && WKUP == 0)
5 ?& l. n, H6 c2 U$ ?; c- i - 37 {, _! n1 ^: C2 i) v6 c4 Z" D- ]# m
- 38 key_up = 1;
+ l* |/ t8 I+ \ - 39 }0 r) G2 ?* S: u
- 403 E' q! E7 a3 _6 A- n' O
- 41 return keyval; /* 返回键值 */
* J1 l9 N# K5 _ z1 c7 Z - 42 }
% T' p! E6 y( P" B: `" g8 b
复制代码
, S: M7 R ]$ xkey.c文件主要是处理按键扫描函数,程序设计按键支持连续按和不支持连续按两种情况。我们来看按键处理的程序设计过程:! S/ _0 v6 g% Y: o) L; s" ?
第19行,key_up为按键松开标志,如果key_up等于1,表示按键已经松开。9 y7 |/ ~; x+ }; U# N6 q( ^
第20行,定义程序的返回值keyval。/ U" \3 A3 o5 y& \6 ^% T& J( N
第22行,如果函数的参数mode等于1,则key_up等于1,表示支持按键可连续按模式,也就是当按键按下不放时,每次调用该函数都会返回键值。如果函数的参数mode等于0,表示不支持连续按,也就是当按键按下不放时,只有第一次调用会返回键值,必须松开按键以后, 再次按下才会返回其键值。0 K' o9 m# R0 g3 _7 M0 G
第24行,如果标志位key_up等于1(表示按键是松开的,没有按下),当KEY0等于0或者KEY1等于0或者WKUP等于1的时候,表示有按键按下了,具体是哪一个按键按下了,需经过后面的程序进行判断。
" T6 w% m+ ^8 \7 P: _: O第26行,程序延时10ms,因为按键按下过程产生抖动,抖动周期大概10ms,所以我们先延时10ms以后再去读取按键对应的IO口的状态。
2 v6 A) C! k" Q- j) Y3 K第27行,将标志位key_up置0,方便程序用于判断下次按键是否有按下。
+ n( N, l, t4 Y# u第29~33行,通过读取到的按键对应的IO口电平来判断是哪一个按键按下了。
; f9 ?' O2 U, p# s% W我们前面通过原理图分析知道,当按键WKUP按下以后,对应的IO口为高电平,当KEY0或KEY1按键按下以后,对应的IO口为低电平。
( g; U$ g# A! x6 I8 S第36行,如果读取到KEY0和KEY1的电平为1,WKUP的电平为0,说明此时没有按键按下, key_up值等于1,表示按键是松开的。+ r" n3 K# v( X* H6 A3 q( Y0 H9 _6 a+ G
5 U" J" ?. R- D/ @. I; q( \6 @8 k$ smain.c文件函数如下:* Q; }7 Y. [/ W4 s7 |
- main.c文件代码) k$ u1 Y ^) u* e5 p- D% L
- 1 #include "main.h"- A U; D9 M* f" F4 T
- 2 #include "gpio.h"( T6 w$ W, L. g8 C. t. R* k7 z
- 3 #include "../BSP/Include/beep.h") l* w* a. F* U* r3 r# D8 ?' A
- 4 #include "../BSP/Include/led.h", w& [8 B' C0 ?
- 5 #include "../BSP/Include/key.h"3 y1 d/ O, s/ q ^
- 6 uint8_t key; A" _* u* i, s" Y6 v
- 7 & {: e3 b& [$ e8 Q
- 8 void SystemClock_Config(void);
9 V& l0 I* @/ c i - 9
/ r4 {# S4 @! M$ E, t - 10 int main(void)) q4 x2 c4 X/ E! I, }
- 11 {
: I# M* J6 H. b( v - 12
( ~: T( `5 L. n& F R - 13 HAL_Init();
& b6 X2 e' @3 f% k( w' G5 R - 14
& N0 O( ~( h3 a8 p$ l# x8 D7 f. g - 15 if(IS_ENGINEERING_BOOT_MODE())
; k. o8 S0 K3 t& t2 K7 u$ C1 C - 16 {# N% `$ E5 Z; d2 u: u. w% y0 P
- 17 /* 配置系统时钟 */$ J s, t8 X: Q+ Z+ ?& [
- 18 SystemClock_Config();% d+ I2 O5 r7 y; `$ |% ?5 x0 A8 W
- 19 }
# s9 m" Y# t2 A! l7 c - 20 /* 初始化所有已经配置的外设 */* E; ?% z" l- z
- 21 MX_GPIO_Init();
5 Z4 O9 u7 t! ]8 f6 r - 22 led_init(); /* 关闭 LED0,打开LED1 */
! M) t, b* j" m& Q - 23 beep_init(); /* 关闭BEEP */
& X2 A' [- l% x, ?- z! U: { - 24
9 X2 T3 M. C+ i' p; Q3 m( F/ ] - 25 while (1)
! D2 f- Y" J) h a* j9 d+ a - 26 {
1 y+ m* b% H6 U% ]! u$ M) } g - 27 key = key_scan(0); /* 得到键值 */ x0 c. B( E1 w
- 28 if (key)! F- S) s5 h8 c# [* V5 V# E" C/ G+ z
- 29 {4 K; @/ ^9 s8 s4 O9 N0 w8 p; I
- 30 switch (key)" H/ p9 |% Q D/ w) s
- 31 {
" K% c$ o8 Z x$ k# G; c7 j+ q - 32 case WKUP_PRES: /* 控制BEEP翻转 */
/ H. `+ z( K" d( J3 k! V0 c6 l - 33 BEEP_TOGGLE(); /* BEEP状态取反 *// t4 r0 B, x% }, Q
- 34 break;. T0 T) N6 \! L$ H
- 35
3 g3 g `# J+ l. D$ \ - 36 case KEY1_PRES: /* 控制LED1翻转 */
- ~5 D$ [, M* k2 z - 37 LED1_TOGGLE(); /* LED1状态取反 */
# L8 S+ k8 D8 T* X, f f( K - 38 break;
" b* A, v+ ?7 {! N+ W5 T - 39
% O+ v: z- T: U! o0 y% A& u - 40 case KEY0_PRES: /* 控制LED0开关 *// _+ E# }! e4 u9 v
- 41 LED0_TOGGLE(); /* LED0状态取反 */
7 C& Z/ o1 k+ [ s - 42 break;- Q4 x$ Q/ T5 [+ c7 X4 X$ K
- 43 }
& `& A8 j( S5 @% N# G4 s - 44 }
+ S; W# z* M6 i/ g6 S5 j - 45 else
& Y7 b c& H' r" [; |2 i5 G$ | - 46 {; ^% x' A5 P- i0 o
- 47 HAL_Delay(10); /* 延时10ms */
5 E# E$ `5 H$ {' S - 48 }
* v' w- E" f5 @. M! | L9 b - 49 }4 v, i: f/ H, c8 b
- 50 }: M% Y( V' y% u$ B! z8 L" A
- 51 /* 系统时钟配置函数 */" ]. _& r1 ^1 I
- 52 void SystemClock_Config(void)2 V8 e9 [7 n3 ~: T
- 53 {
6 l$ v+ E2 C' J# O+ | - 54 /*此处省略时钟初始化代码*/5 |9 b4 Q8 f& | F8 l
- 55 } f: l* l( ^6 f5 h7 Q
- 56
6 ~% U0 `8 b: [ - 57 void Error_Handler(void)8 }. a6 q# p8 Z# q* h0 u# K
- 58 {
$ d( h9 h8 A' j3 u' t - 59 5 n: C$ p* v( t, t
- 60 }
复制代码 ' H; L$ m2 j7 B- z! g
main.c文件的SystemClock_Config函数我们前面的第九章的实验中有分析过,因为我们前面没有去配置系统时钟,此时时钟源默认采用高速内部时钟HSI(64MHz)。MX_GPIO_Init# I+ P$ [' O- [: A
6 M; K. g- ^8 U+ ^: Z( @
函数是GPIO初始化部分,主要打开用到的GPIO外设时钟,并初始化用到的GPIO的工作模式,我们在前面第10.5.5小节有分析过。3 a {5 M1 R M9 K5 X4 n7 |
第22和23行,在进入while循环前,LED0和蜂鸣器是关闭的,LED1是打开的,所以后面Debug运行的时候,会看到LED1是亮的,LED0不亮,蜂鸣器也不会响。
3 i! ^9 ]$ ~/ v" ]6 n第25~50行,while循环处理的过程。
0 R0 m6 T9 U) z# k/ \第27行,先获取按键的值,这里key_scan(0)函数的参数是0,表示不支持连续按模式,如果大家想测试连续按模式,只需要把函数的参数0改为1即可。9 U+ w1 ]7 q B$ |
第28行,key_scan(0)的返回值key可以是0、1、2或者3中的某一个,如果key的值是1、2和3的话,则表示有按键按下,通过第30~43行的case语句判断按键的按下情况:4 P, }) r/ Q( i# p" b$ t
第一次WKUP按键按下,则蜂鸣器响,再次按下,蜂鸣器不响,再次按下蜂鸣器又响,如此循环,实现蜂鸣器翻转。' d% n- y0 i7 N/ h) z& p& Y4 @1 l
第一次按下KEY1,LED1灭,再次按下KEY1,LED1又亮,如此循环。
' @1 Q5 ~' L9 f/ g0 }( D: m; u' R第一次按下KEY0,LED0亮,再次按下KEY0,LED0灭,如此循环。
4 c4 a3 i/ C" g8 X如果key的值是0,表示没有按键按下,程序执行HAL_Delay(10)函数延时10ms以后再返回到第27行往下执行,即在while中循环。3 E: O3 r v- _6 F8 r
, G) |. o7 @) a" I( t+ p; G12.3.6 编译和调试. B- @. }4 Y$ g j" V
按下“Ctrl+B”编译工程,按照我们前面的操作进入Debug模式,然后点击运行按钮来运行调试,可以看到开发板底板的LED1 灯第亮的,LED0不亮,蜂鸣器也不响。然后开始按下按键来测试实验结果,实现现象和我们上面分析的一致。% f* o1 j/ p2 g
6 s+ S( D! f7 t* V1 D) h; P12.4 章节小结
, c& u8 L5 @0 c4 `$ s经过前面的实验学习,相信会有部分小伙伴对什么时候软件要配置上拉什么时候软件要配置下拉有疑惑,本小节我们就来总结这部分内容吧。; X! Q8 g# J! W
随着技术的迅速发展,芯片的集成度和复杂度也越来越高,功能也越来越强大,现在的芯片内部一般都集成了电阻和电容,为PCB设计极大地节省了空间。如下图所示,STM32的GPIO口内部自带了上拉和下拉电阻,电阻上有一个开关,可通过软件配置来设置开或者关。
! v7 `, A7 v6 C. T: |9 e8 L
5 Y( c. Q3 ~" e) e
" v1 a& x; I! ?! y' q( G
7 z# P0 X' b5 @图12.4. 1 GPIO的基本结构图
/ Z! O2 ^9 f- V, a! `上拉就是将一个不确定的信号通过一个电阻嵌位在高电平,电阻同时起限流作用,下拉就是将一个不确定的信号通过一个电阻嵌位在低电平,电阻同时起限流作用。STM32的这个上下拉电阻阻值一般是30KΩ~50KΩ,上拉电阻,一般称为弱上拉,下拉电阻,一般称为弱下拉。这里的弱是指电阻阻值比较大,电流就比较小,充电就比较慢,从而上下拉的速度就比较慢。
+ F; _0 O3 |2 T7 ~( G* _例如,芯片内部开启了弱上拉(例如阻值是50KΩ),外部再加一个强上拉电阻10KΩ,电路中这两个电阻相当于并联,IO口的上拉电阻就小于10KΩ,电流就变大了,也就变“强”了。此电流可以供开漏电路使用,开漏输出最主要的特性就是高电平没有驱动能力,需要借助外部上拉电阻才能真正输出高电平。
/ P3 ^, m) u! r/ C' L芯片复位以后,端口上拉下拉寄存器(PUPDR)的复位值为0x0000 0000,即默认不开启内部上拉和下拉,IO口相当于浮空状态,电平状态是高还是低不确定。那么问题来了,如果此IO口接的是一个按键,如果外部不做上下拉会怎样?我们来分析前面的实验。
% n0 X |( w5 z0 ^9 y* T
1 D( m; |- D* M
$ k4 J9 Q- c- B' \' P: i
9 ^4 Y! P% L+ V4 s图12.4. 2按键原理图& q4 D4 B9 J) R3 H. E
我们是通过读取按键对应的IO电平状态来判断按键是否有按下的,按键WK_UP一端接的高电平,另一端接的PA0,按键按下以后IO口是高电平,所以认为此按键是高电平有效。如果PA0内部不开启下拉,那么IO口电平是未知的,当按键程序去读取IO口电平时可能读到值为0也可能为1,如果此时恰好没有按下按键,而程序读取IO口电平为1,程序就认为按键是已经按下了,这就错了。在按键未按下前,我们当然希望PA0电平为0,所以对于按键WK_UP一定要开启弱下拉。对于按键KEY0和KEY1是低电平有效,IO口的一端实际上是已经接了10KΩ的上拉电阻,IO口电平已经默认为高电平了,所以内部上拉不开启也是不影响的。
9 u1 ^& W2 P5 K1 J& M上下拉电阻的目的是给IO口设定一个确定的电平状态,如果按键按下是高电平有效,我们就做下拉,让该IO口在默认状态下处于低电平状态,即没有按键按下时,IO口检测到的总是低电平,只有按键按下的时候IO口才会检测到高电平;如果按键按下是低电平有效的话,我们就上拉,让该IO口在默认状态下处于高电平,即没有按键按下时,IO口检测到的总是高电平,只有按键按下的时候IO口才会检测到低电平。
8 X: N: B! r9 A) Q————————————————8 i7 U/ w% e7 p# B& q8 E1 K
版权声明:正点原子
* X* h* q6 _' w' \$ W/ |
# r x! h$ h w5 w; Y. F* E+ p# {2 S# \6 ^/ E9 U9 A2 A0 z
|