请选择 进入手机版 | 继续访问电脑版

你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

学习STM32单片机,绕不开的串口

[复制链接]
STMCU小助手 发布时间:2021-8-9 14:32
    刚开始学单片机的你,是不是会因用程序把LED点亮而感到高兴,会因用程序把数码管点亮而感到高兴。这是好事,这也是想继续学习下去的动力。
+ e$ J1 ?6 O/ d2 D- j5 W0 x+ l) C8 p8 y

# X9 T5 r1 H9 B1 M) w* b    但是数据相关的实验是学习单片机和STM32的一道坎,此时就需要用好本文要说的串口这个调试工具。) x: R5 a0 |& I: |! o% b

* l- g5 \; s4 `; [+ W
( \" v7 u5 H. |9 Q
串口通信介绍6 o4 t" v0 A3 e# B

3 K& e0 i: z: m% m) _

. J, z# O. z2 l$ \1 O$ b3 l5 M' W3 Z    关于串口通信的介绍,也可以点击阅读之前发的一篇文章:  STM32串口通信基本原理4 ?) n9 a  S$ `) D; F4 Y8 t
! t2 ?* s. s0 p& A, H2 i

1 E6 c3 q) ]: q- `. U8 Q- i! a    串口通信是指外设和计算机间,通过数据信号线 、地线、控制线等,按位进行传输数据的一种通讯方式......这种太过理论了,看似懂了,但又不懂。还是用我笔者自己的话来说吧。8 P  @/ R- M/ d5 M: e' \; ~
. R+ s/ E, Q' K
4 v+ f, c. G( {! x" F1 m! W3 Y$ V6 ?
    串口通信就是可以把程序在单片机或者STM32芯片中运行的结果发送到电脑的一种通信方式。. T3 a1 y9 q7 f3 I5 B

) q$ s" ]! R* d! d7 K3 ^1 k
& l( {6 v* X0 a: P
1.png

: A* o9 k) G) n2 g" [7 d& |1 j0 w, r- Y# k4 g% _* Q9 j
7 f6 d( f  n) x2 x! h1 u( V
2.png
5 E0 c4 [! p- `/ [( q

) ]+ X" U8 n' U$ a: v

  F3 l- i! L7 L: g2 S* v    如何使用串口通讯,你需要知道的几个重要的知识点:( Y9 u. v9 x8 v) i( K7 v- M" w6 O+ I
+ ^1 t& l+ |' k0 G

! ?- S/ W$ f$ y3 d
  • 波特率,数据传输的速度,如上图设置为9600,那么程序中也要设置为9600
  • 硬件连接,接收、发送交叉连接,如上图TxD->RxD,RxD->TxD
  • 停止位
  • 奇偶校验
  • 硬件数据流& ]& M% i$ j9 O! r) k1 h

0 J% e0 i; X' s4 Z% N
% F) M/ r1 J: ?/ r1 c# S" X; v
, c1 I2 r( a$ T' p6 l5 h/ N
    其中,后3项一般不需要改变。+ `% a' S2 C7 s1 W2 a4 {; N
. Q( G  r- X2 j" u" a' q
" G0 x9 X" @* B! ~0 m" B0 j
学会串口通信能做什么
: w- K' [2 \* ^7 C. ?. J  E0 G+ P/ Q

) s* q& X# C9 B+ {1 _+ h    学过C语言的人都知道,程序开发需要不断调试不断验证。很多的语言编程软件都有很完整的程序调试功能。使用起来很方便。而我们的开发虽然也是用C语言,但我们的程序最终运行的环境不是在我们的开平台而是在真正的硬件系统中运行。这时想要查看程序的运行过程或者结果就不像编程软件那么方便了。
* j" }9 z! T9 T9 x
6 Y% @; O* v5 A6 ~! r
* w. O+ {2 O( A8 l) F% s1 P9 I
    想要解决这个问题,我们这时就可以借助串口通信来把我们需要知道的结果又或者是程序运行的关键步骤发送到电脑上,我们就知道程序在硬件系统中运行是否出现问题。51单片机可以用串口来调试,STM32可以用串口来调试,Linux开发板也是使用串口来调试。0 U* Q/ o2 D& ^* y! w( r2 X  S

' D+ N) k- O6 l' ?/ @
& r2 B; [5 _  @3 c
    当你学会串口通信时,你可以开始玩WIFI模块、GSM模块、蓝牙模块、GPS模块、以及各种使用串口通信的传感器等等。有能力你还可以编写上位机软件通过串口通信来控制设备。
# i/ n: q' P* D, Z# ]1 `9 n% h5 |1 ]6 n0 N; f

$ s: m6 o& w6 E2 g4 E9 Y串口通信需要什么
3 ~" M. I9 c6 y/ i  l; K) Y) I
5 q' T& D5 h1 A5 H& s3 ]0 ~' ]

6 X" u6 z8 S( m    如果你只有STM32核心板,那么你还需要一个串口转USB模块和一个串口数据接收软件还有几根杜邦线就可以了。
  ?0 N5 e' y$ D. y! o. \# r( T% t9 [" Y- k9 J1 P& I
9 C/ ^* Y7 O) Z% M; I/ o! U& }8 ?
    如果你的是比较完整的一款开发板的话,一般都已经带有串口转USB模块,这样使用起来就更简单了。聪明的你是不是发现还缺少了点什么,对。没错因为还缺少了最重要的程序。想要使用串口通信当然还需要写串口通信的程序。
3 g8 {$ B  G: B; m& h
0 S1 S5 z4 g9 f! a2 ~% l
9 G4 i9 h) p3 U
    下文就教你如何去用,而不是写。
; k9 K$ O: `; f/ I7 S4 _- W2 F
# e' u9 h5 j6 K
, P2 c, t- T- e4 `
串口实验
! i9 A# x4 N! n- _# Z, z
0 u% b- R' c; ~! L3 m0 E

! t! _. B# F* O7 K6 h! z! ?: B; }    关于串口的实验,之前录过一个基于HAL库来实现的视频,可以点击查看。; I3 E' M6 Z: l0 H

5 f% z, b2 K+ d: T$ F
2 i9 j# M: c0 L3 K% S% p
    STM32打印数据到串口助手4 X$ `+ v1 M1 X  I$ Z$ s- o4 B. W

5 ?- O1 ~# c  G" M& f
& |- X2 o4 D5 v1 K; Y$ w
    而下文中的实验基于标准库实现的,与HAL相比思路是一样的。0 A$ l9 r8 l7 U0 g+ X/ `/ I* v

% E+ ^4 e7 ]" v% t' I% w: p
6 Q& A) k7 f* s# p: [1 ?
    在做一个实验时,最好把这个实验分割成几个关键的步骤,这样做的好处就是可以清晰的知道自己需要做什么,以及做完了哪些。还有哪些还没做。
4 x2 U  @# R/ @4 ?+ z, p$ m- t: O. H% N& @! v7 r/ o- Y6 W  w

; q# |' q$ h( l" z, k0 Y    下面把串口实验分成几个关键的步骤:
' t) o1 F% Z. b, {. _1 c! s1)串口通信使用到的GPIO引脚配置# V  |! P" e# O$ U* e
    STM32F103系列的芯片一般都有三个串口以上,用来调试使用的串口一般都是使用USART1。其他的串口配置都是一样的。
; n: J7 r! \* w3 U' _4 d2 w4 i- b* T+ E
/ k( h, O, m; j) m9 D/ v* |6 v
    下面这段就是串口配置的程序:: G9 W" j5 n# }  Y5 S2 Z( C5 {; ~
  1. <font face="微软雅黑" size="3">GPIO_InitTypeDef GPIO_InitStructure;5 |9 F/ B5 c4 E

  2. & f+ o) g* u7 w" b3 v
  3. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);  //使能USART1,GPIOA时钟6 ^' B6 J" k' [; T; K0 B

  4. ! V4 ?7 Y. X  B- p. w
  5. //USART1_TX GPIOA.9
    % v3 `8 n  J* @) D" W
  6. , }& d4 t% A& y  Y/ G1 x9 O1 m
  7. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
    ' P8 ]0 }1 W# o: x- }
  8. ' L7 ^" K! `9 j4 j+ g9 c
  9. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;0 _# I5 b3 W! y. S; I& H5 g, R3 n
  10. & ~  h" X! i+ l7 Y
  11. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  //复用推挽输出' O& C2 ]( B: V. e7 u

  12. ; B* r- n( U& J! F! p' {
  13. GPIO_Init(GPIOA, &GPIO_InitStructure);  //初始化GPIOA.9# x1 T. q1 J) O8 A+ x5 N5 [  D

  14. * v* X9 ?% I2 u3 \2 k2 {
  15. //USART1_RX GPIOA.10初始化' j  s  j. n. A, x9 a! _' `, Q! e

  16. . J* z: M: L, H0 R
  17. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;  //PA10
    2 {' y/ t: R" R: R! i* i" M

  18. # P! E& K! L8 s- O8 v: X+ z
  19. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;  //浮空输入4 e9 R5 i# V4 t) m& X  i4 K2 D2 C2 q

  20. 4 i) l# i+ Y3 |  E) M7 e  P
  21. GPIO_Init(GPIOA, &GPIO_InitStructure);  //初始化GPIOA.10 </font>
复制代码

4 a2 u+ i3 h. d7 M

+ E: S, P8 e+ G    串口使用的的GPIO口是PA9和PA10,所以只需配置这两个IO口的输入输出模式就可以了。
' A* L( N8 s  P- O5 u3 ~$ B0 v" z( y" x4 U4 Z

6 K4 N) u4 q# n& _7 V2)串口主要参数设置(直接看程序)4 h9 W1 j+ x  w: N+ ?- {

% ]  v) c3 g- x$ o% \

' U$ t; r  s7 ~. ?0 ^& \
  1. <font face="微软雅黑" size="3">USART_InitTypeDef USART_InitStructure;# f8 _8 C4 O1 Q3 C' [* a

  2. 5 r" ]2 t0 j" L  x/ {" S% B
  3. //USART 初始化设置# E4 N6 q, k: D7 n- Z

  4. 7 p5 S  S+ g( ]9 {
  5. USART_InitStructure.USART_BaudRate = bound;( K! q- j3 `: g$ q1 N% C
  6. : f/ u6 L; ?  q& V5 _' ?, B
  7. //串口波特率
    & J% S& `  \5 k/ t& o3 t& o

  8. ; s% M9 X. w" r) u' w9 J$ y
  9. USART_InitStructure.USART_WordLength = USART_WordLength_8b;: h0 O8 j1 F2 N8 l; q

  10. 4 I5 I: N7 |- |% Z. x
  11. //数据格式,8位
    , f- V$ k2 ^" M, q$ E

  12. 3 Y: X- _! n& x. U$ i, Y
  13. USART_InitStructure.USART_StopBits = USART_StopBits_1;+ r$ ?  \1 h1 |
  14. 9 \* h+ G8 q/ o
  15. //一个停止位+ E5 v2 T6 E# I3 ]: C
  16. / N' [* ?1 X1 M6 ?/ {% x, T1 C
  17. USART_InitStructure.USART_Parity = USART_Parity_No;  p! a. f7 R( Q; R! T
  18. , q; l9 g/ p# n5 y  E4 c; S0 S  b& v
  19. //无奇偶校验位6 e* A6 i& T8 F6 O. U) K1 K

  20. 0 E! ?5 A: s3 F8 P' h) g
  21. USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;7 n# Z- Y6 A- o% u- ]; b
  22. , a& j3 [7 R! z: z
  23. //无硬件数据流控制
    ! ?# N# ^3 j2 B4 U" F0 T- u" D( x

  24. 3 b. \* L- {- {$ P/ T- n. e) f. r( S- o
  25. USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
    . J( t/ N; G3 G; |
  26. / z& P  I  L4 |$ ^4 d/ L7 p" N" s
  27. //收发模式
    9 e3 |) s5 n0 M7 `. z
  28. 2 F! f+ l# V- G% Z9 E
  29. USART_Init(USART1, &USART_InitStructure); 0 e) ^& F2 D' y% g

  30.   x$ ~$ H& p; }- j. V/ l
  31. //初始化串口1% K# ~5 a. S$ q! k

  32. ; d7 F; P- P" A( L4 J2 Y* o0 z/ o& O% x
  33. USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);
    8 r8 p( m1 T2 M" g/ h. J+ s

  34. 7 G+ J- q; C+ J8 t9 d
  35. //开启串口接受中断</font>
复制代码

0 u8 V' g. k& n3 R5 N
' W0 p$ h4 j$ k% w+ D1 R
    串口参数配置无法就是配置串口的波特率、数据格式、停止位、奇偶校验、硬件流、收发模式。除了波特率需要改变其他的参数都不需要管。直接复制拿来用。
: v7 z7 J7 J5 p
) z, l5 y5 i  J9 ?/ V) Y# c# |
+ I- k5 g; m. @1 D) a- D
3)串口中断配置: v0 d1 w7 O, f9 E. O$ J( ~$ K
    串口如果使用中断接收,那么就需要配置串口的中断参数,配置项无法就是配置那个的中断源和中断的优先级。8 C) k3 l& q6 a6 }/ u; u* p; p

; J- @9 M( O0 l; l8 s  W+ P" n9 n& S5 A
" ?: S) \$ ?( T  c8 o) K3 o( n
  1. <font face="微软雅黑" size="3">NVIC_InitTypeDef NVIC_InitStructure;* }  E) h; `" z2 U& A7 s) i

  2. ( e3 |' Y6 M8 L  A2 O$ g0 o0 {2 L
  3. //Usart1 NVIC 配置/ x4 U2 u$ W! D1 m( s0 i% f

  4. . g  w" ]1 @% X8 p4 t9 s
  5. NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
    % s+ x7 P+ h& B( U% [3 Z: g

  6. ( Y# j* z* p1 K5 E2 o2 x
  7. NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;  //抢占优先级3
    ! Y. b3 S, E+ t/ Y3 C
  8. ( v9 g) B+ B" R% n/ Z/ h5 x
  9. NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;  //子优先级3
      P- f9 y$ v1 Q- y+ s4 _
  10. 6 K3 X& W7 X* F( j: |$ S
  11. NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  //IRQ通道使能
    ( J% a' m; Q5 @0 U  B) v. E

  12. . X: I, _1 l) p& c+ }! w8 g  ]% Y
  13. NVIC_Init(&NVIC_InitStructure);   //根据指定的参数初始化VIC寄存器</font>
复制代码

; B7 Y. A7 s! @
& t9 H# `0 a) W7 q4 F/ K
4)串口使能
! |+ C7 X) \+ `; W    就是你需要什么时候开始使用串口功能,就是一句原有的函数。
( `' |% M- ?; N- W
  1. <font face="微软雅黑" size="3">USART_Cmd(USART1, ENABLE);  //使能串口1 </font>
复制代码
, z* v, `, ]' y6 |& i. P0 d, Z
+ u9 g1 Z$ Y1 j. ]2 O
5)编写串口中断处理函数: R/ G, _+ f! C& ~; }9 T
    使用库函数开发,所有的中断函数都是已经存在的,只是中断函数里面没有处理任何事情而已。中断函数如下:* l# m4 f( ^0 n; W5 s1 u, g

- \8 P: f2 V' i; k6 [

6 M9 e4 |0 r6 j! _ 3.png
5 d$ ~. E, P3 M  B* p* ~. f9 g! A' f4 @9 H) N( i

/ Q( _: K# \7 F7 j5 ^% X& I    完整的串口中断函数:
5 m' p* J0 p9 t! K4 J
  1. <font face="微软雅黑" size="3">void USART1_IRQHandler(void)
    ) O! S# \  `: ^6 `. r" p

  2. " ?4 [& V8 d& L7 D. e- `& i
  3. {
    ' E9 I: q/ B! V+ r, B
  4. & }. ^8 k5 T( a3 o8 p; o) E% ^
  5. int Res=0;  //定义一个变量用来接收串口数据集% F1 {- ~; x3 x/ y( l1 G, N  U. X
  6. . \0 D$ W! \3 {- Z" D+ z0 v
  7. if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中断(接收到的数据必须是0x0d 0x0a结尾); @. [5 q2 U; _& x1 t

  8. ) \" X7 g9 Q5 K4 \- g
  9.   {& O+ f8 s- M; h  b

  10. * h) i' _/ O8 I/ B7 B
  11.   Res =USART_ReceiveData(USART1);  //读取接收到的数据
    ' [4 ]6 M8 h: t
  12. # ?# v$ @8 W( d% v$ P4 q
  13.   USART_SendData(USART1,Res );     //把接收到的数据通过串口1发送出去( x  `$ q0 c& Q

  14.   s5 i( U; h9 f1 ~' q
  15.   }
    7 p  _3 B4 p; M0 t9 X( Y

  16. 9 k  o8 w; Y7 A6 z$ Y+ X  l" B
  17. }</font>
复制代码

; h, r5 B; E3 R; Q

2 I+ C' x, `1 f) U/ L    串口数据的接收和发送的函数都是库函数提供的,想用时只需找到它直接拿来用就可以了。6 p; }" ~; {. a3 ~4 K8 G
% }6 _' r( c& A* d$ l

1 A, w7 Q) ]' Y: W& W; B6)实验现象" L" z, g3 ~8 d' Z1 h
6 S) W$ S, i. e

; l  `) {$ h2 _0 m! b3 B    把程序编译烧写到STM32然后用串口转USB模块连接到电脑,在串口调试工具设置好波特率,打开串口。正常的话,那发送什么到STM32单片机,串口工具上就会收到什么。
+ r' \; x. V; V! m, A$ u+ E
! P8 U2 E) f2 D* x

+ H- Z& O& `, t6 p7 m& g' @总结/ s/ N8 p1 B% j! v! F8 E, q
9 D* c. Z; b* k! n$ ]1 l. i1 q: d

" n9 ~& `# S  B: j- c, y    以上就完成了一个最简单的串口实验。
# W) F7 A$ N4 B$ m4 V
, V: a9 U; U' J+ z" ]; Z
8 e% y0 \& U, E& Z1 n! D' P( Z
    本文章主要是说明串口的重要性,用来调试程序很方便。
: q- J  y' S' I9 y' |
. x' k9 ?" v# y. c
收藏 评论0 发布时间:2021-8-9 14:32

举报

0个回答
关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版