HC-05蓝牙串口通信模块应该是使用最广泛的一种蓝牙模块之一了。为什么呢? * v( }: ^2 I6 G! {4 r% _8 N
因为HC05模块是一款高性能主从一体蓝牙串口模块,可以不用知道太多蓝牙相关知识就可以很好的上手。说白了,只是个蓝牙转串口的设备,你只要知道串口怎么编程使用,就可以了,实现了所谓的透明传输。
$ ?' R8 N9 c0 p" x$ B1 C( `$ f1 ]
但是就是这么一个很常见的模块,网上很多的博客写的都是错的,或者都是很不详细的。 1 |: { F; R, X# J7 E
所以本文就介绍一下这款蓝牙通信模块的使用,包括蓝牙模块的调试、手机与蓝牙模块之间的传输、手机蓝牙控制STM32单片机,应该是逐渐深入的一个过程。但是这仅仅是使用,以后有时间应该会对蓝牙有一个稍微深度的学习,而不能仅仅是浮于表面,只会用。
0 f# Y; v! b% y) F. i c: s模块名称:HC-05蓝牙串口通信模块 其他模块:USB转TTL模块、手机蓝牙串口助手app 手机蓝牙串口助手软件,可以点击链接下载:蓝牙串口。因为这是我见过所有手机端界面最好看的了,其他的界面都有点太糟糕了。
0 L" A: ?! s" T: f4 S7 p蓝牙模块的调试准备工作USB转TTL模块与HC-05蓝牙模块的接线: 两模块共地,两模块共VCC(VCC取5V);蓝牙模块的RX接转换模块的TX,蓝牙模块的TX接转换模块的RX。如下图所示:
% B- F- y9 W0 ~* [4 Y

( n/ y) Y6 X$ ^: ]
这个时候就要将转换模块连接到电脑上,然后利用串口调试助手进行蓝牙模块的调试。
6 E( j( h- A, {, k# O, d蓝牙模块的调试4 a! S( |( z4 n
HC-05蓝牙串口通讯模块具有两种工作模式:命令响应工作模式和自动连接工作模式。在自动连接工作模式下模块又可分为主(Master)、从(Slave)和回环(Loopback)三种工作角色。
+ Q- z% ]8 R* r8 n2 `
- 当模块处于自动连接工作模式时,将自动根据事先设定的方式连接的数据传输;
- 当模块处于命令响应工作模式时能执行AT命令,用户可向模块发送各种AT 指令,为模块设定控制参数或发布控制命令。- B- s! x* n7 l i
7 x8 g7 U. K2 y
怎么进入命令响应工作模式? 9 i2 H; u$ P) ^ r* g. g! T
进入命令响应工作模式有两种方法:
1 B: H R6 ]6 Q- J* e+ Y
- 模块上电,未配对情况下就是AT模式,波特率为模块本身的波特率,默认:9600,发送一次AT指令时需要置高一次PIO11;
- PIO11 置高电平后,再给模块上电,此时模块进入AT 模式,波特率固定为:38400,可以直接发送AT指令。
3 v0 h5 u- ~8 _6 m" O5 Y
4 m ?* d; m, Y2 X
M5 B, B- O, u9 C0 X( x/ H y/ N; Y' E
什么叫做置高一次PIO11? 5 U) V9 g; r5 w. \ S! G- Z5 }
在蓝牙模块中有一个小按键,按一下就置高一次PIO11。也就是说,第一种方法需要每发送一次AT指令按一次;而第二种方式是长按的过程中上电,之后就无需再管了,直接发送AT命令即可。 ; s9 g# G" O* j/ _# \$ n
需要注意一下,两种进入命令响应工作模式的方式使用的波特率是不一样的,建议使用第二种方式。
2 ^- S, ]1 x$ h1 m& v/ ^9 F/ b
怎么区分进了命令响应工作模式呢? 0 P5 }, `/ @" a) }
在蓝牙模块上有灯,当灯快闪的时候,就是自动连接工作模式;当灯慢闪的时候,就是命令响应工作模式。 5 U* Y$ V3 g2 M( o$ ]" O1 o
AT命令/ ?$ O( q4 i& I/ c) V9 R
进入到命令响应工作模式之后,就可以使用串口调试助手进行蓝牙调试了。
2 X [9 c. _5 e
首先有一点,AT指令不区分大小写,均以回车、换行结尾。下面介绍常用的AT指令:
; e, }( X$ x I) R H: k4 t. b/ v; J6 c: N
- K3 W8 P/ a3 @5 K$ Y
对于AT指令,有几点注意: 6 g" `8 a. C( z9 E+ `7 B! z
- AT+NAME?:获得设备名称,这个AT指令有很大可能性是没有返回的,因为我也看到了很多的例子……,但是其他的指令都是没有问题的,直接设置设备名称就行了;
: i' F) d$ M; q+ V
, m2 j5 E n& v$ f" a/ [6 p+ |1 v
( M! X" d. l% |# K# e; k7 \- AT+UART?:获得串口参数,串口的参数一共有三个,波特率、停止位、检验位。其取值如下:, R6 T& ~/ [1 W- o7 q+ l. P
: b5 i$ _' D8 V! D/ {3 F
; X) b0 h5 P4 m8 K/ [. j; x- T& c- l; a! N3 b3 E1 P
串口参数9 }5 a0 z9 F! ?8 S4 C
4 o+ [% g. K! K& [/ E) x, |" a
参数名称 | 取值 | 波特率 | 2400、4800、9600、19200、38400、5760、 115200、230400、460800、921600、1382400 | 停止位 | 0:1位 1:2位 | 校验位 | 0:NONE 1:Odd 2:Even |
2 _7 @' ^, ^ k+ r" ~ R: g其默认值为:9600,0,0。
, B: t; V' }5 m" p0 i5 w- z
例子: 
c3 g2 d; W5 E- z. H* _9 ^
本文中,蓝牙串口的波特率设置成115200。之后的内容,就会采用这个波特率来进行通讯了。
9 k- K* S: K: s$ _1 t2 b* t手机与蓝牙模块之间的传输
9 h0 a5 G" o! I直接将蓝牙模块与转换模块连接,再讲其连接到电脑上,蓝牙模块直接进入自动连接工作模式。
9 j, |: U+ p3 ]+ ]1 Z
此时手机打开蓝牙串口调试应用,用其来连接蓝牙模块。手机蓝牙串口助手软件,可以点击链接下载:蓝牙串口。万分推荐这款,因为界面脱离了那种黑不溜秋的感觉,比较简洁、清爽。
0 E% G/ F6 \& V3 f5 o0 o
这个软件的使用:点击界面右下角蓝牙的标志,选择蓝牙进行连接。 ; i: e! D+ I8 Q% J9 W; ~# M
然后在电脑上的调试助手和手机的蓝牙串口调试应用之间就可以相互传输了,比如: - o4 b) y8 g1 d- h! t2 w

) T0 ~. I, o5 B* R0 C; q8 ], `6 r

1 p8 D1 ^+ F3 M) s3 N5 [* M8 l
可以清楚的看到:电脑向手机发送了“hello you”,手机向电脑发送了“hello world”。
$ }3 K5 z3 R2 {* G& T手机蓝牙控制STM32单片机; O5 ~+ f: z& P( }* z! h0 _
之前的两个例子都是相比较而言比较简单的,这个例子将会涉及到程序的内容了。 7 g' m5 X5 c, \& d4 O5 S
, m7 X6 Z1 O! U; l! w) M; j
实现功能:手机通过蓝牙,向STM32单片机发送消息,STM32接收到消息之后原封不动的返回给手机。当然如果掌握了这个例子,也可以修改成,手机发送特定的消息,然后,STM32单片机做出相对应的动作。比如:点亮LED等、发动电机等等。
. r+ N+ C2 h2 P8 Z l5 K3 s 连接说明
, l* `7 W+ |# X! i s7 {
使用USART1进行试验,也就是说STM32选取PA9、PA10来和HC-05进行连接。同时手机通过蓝牙来和HC-05进行连接。
' O/ G" m; }# f' i' f0 C) O
原理就是:手机通过蓝牙传输到HC-05上,再通过串口通信和STM32通信;而之前一般都是电脑上通过USB线转串口的方式,通过串口和STM32通信。本质上没有区别的。
# b2 p7 k/ @1 M: N1 H: ]
这个时候就应该更加深刻地体会到了本文开篇的一句话:说白了,只是个蓝牙转串口的设备,你只要知道串口怎么编程使用,就可以了,实现了所谓的透明传输。蓝牙的相关一切都被封装起来了,都不需要接触到。
2 Q. x; g) v3 g6 t: [. K STM32控制程序
/ Q8 a' l8 I. e* [* P- e9 N$ W, O0 Z
- #include "stm32f10x.h"
+ P, M- [/ l6 l% T! n. P - 7 W: s3 E* b. }- [% |6 c5 C/ R
- void My_USART1_Init(void) ) h6 t4 b! I/ e E! J
- {
8 T7 Y$ I" @& K8 S" H& \" E9 a - GPIO_InitTypeDef GPIO_InitStrue;
k, Z1 z# w$ v0 ?0 t3 s* ]# e/ p - USART_InitTypeDef USART_InitStrue; * q, K9 H3 w" [+ R3 y% b, A
- NVIC_InitTypeDef NVIC_InitStrue; _4 y) }2 k* I7 \
- ( J. y2 L7 v' g7 c8 W7 o4 P
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//GPIO端口使能 5 Y k& l& P2 p" _/ W+ M
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//串口端口使能 : s1 z3 O. Z0 ^6 ~# N5 n1 f
-
" i$ O2 M4 o: O: [& i( d - GPIO_InitStrue.GPIO_Mode=GPIO_Mode_AF_PP; 8 [! t1 a0 [4 A
- GPIO_InitStrue.GPIO_Pin=GPIO_Pin_9;
: N+ j6 {5 H2 D- d9 H# B - GPIO_InitStrue.GPIO_Speed=GPIO_Speed_10MHz;
4 e! b% i' @$ m9 b - GPIO_Init(GPIOA,&GPIO_InitStrue);
' M& a! ?. _5 N5 z+ z -
! D4 j" r/ E; m$ I( g9 ] - GPIO_InitStrue.GPIO_Mode=GPIO_Mode_IN_FLOATING; ) m8 s) y+ |' ^
- GPIO_InitStrue.GPIO_Pin=GPIO_Pin_10;
k/ r/ Q1 U: z- Y5 Q" }+ n* Z - GPIO_InitStrue.GPIO_Speed=GPIO_Speed_10MHz; 3 d, |+ o; V, r& H: S! b
- GPIO_Init(GPIOA,&GPIO_InitStrue); + V! g: ?: ~- }; {. l$ D, t
- 5 V# w/ N- r* E5 F
- USART_InitStrue.USART_BaudRate=115200; + E- Y, N: g' I7 M1 I3 i/ Y ^
- USART_InitStrue.USART_HardwareFlowControl=USART_HardwareFlowControl_None;
6 M9 \" b, Z3 n/ q* {. y& m$ N - USART_InitStrue.USART_Mode=USART_Mode_Tx|USART_Mode_Rx; * P; Q2 \$ t3 Q
- USART_InitStrue.USART_Parity=USART_Parity_No; & o! M! N$ g, k3 I# w& E7 f
- USART_InitStrue.USART_StopBits=USART_StopBits_1;
+ K% y6 n# I% f1 ~& @ - USART_InitStrue.USART_WordLength=USART_WordLength_8b; % W* @+ k1 I0 P) R' t+ w( n
- , t; s1 ^6 N' K; w N0 h
- USART_Init(USART1,&USART_InitStrue);' j _0 ~* D: h; B; M$ j1 v# }
- " M. v% @4 N$ K+ z/ r+ r
- USART_Cmd(USART1,ENABLE);//使能串口1
1 [. ^, T7 B3 i, U -
( D3 c5 n6 ?: E - USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//开启接收中断 ' P! P% q& X5 R* x: Q& K
- 5 _3 I+ K; X: z# e3 n' l1 ~' h; B
- NVIC_InitStrue.NVIC_IRQChannel=USART1_IRQn; 9 v6 E9 Q) `1 Z! N$ G
- NVIC_InitStrue.NVIC_IRQChannelCmd=ENABLE; 9 n9 f, q1 ?6 I, N
- NVIC_InitStrue.NVIC_IRQChannelPreemptionPriority=1;
+ J$ {/ d% z: Q& B - NVIC_InitStrue.NVIC_IRQChannelSubPriority=1;
% n# R8 k3 C1 H0 H! n6 U" G - NVIC_Init(&NVIC_InitStrue);
$ F- R* \) s# s5 M4 x6 H: ] -
6 z6 _0 R" i/ u2 R2 L4 D& u0 k - }
& Y* K0 K, s1 |/ M/ u - / Y$ I6 V$ ?3 ]" Z
- void USART1_IRQHandler(void)
3 E6 g) d- `5 H& b6 I4 ? - {
6 ?- n: D; u, |& d) W - u8 res; " F+ ^! y$ I- N4 `3 W0 C
- if(USART_GetITStatus(USART1,USART_IT_RXNE)!=RESET) $ o; i$ h9 Y" _+ P1 K) r' [
- { 4 Q0 b% ?6 ~. P# Q* R6 h9 v/ g# {
- res= USART_ReceiveData(USART1); 1 ?* ~ E3 o; R
- USART_SendData(USART1,res); ( U. u, \4 z2 v
- }
b- u7 D9 s9 l - }
( S- `( }8 o1 B, E9 D% y* c5 q0 f1 k -
; b$ `0 I' T- L# `4 i% p - int main(void) + Q) c+ D% N; e2 I
- { ( Z0 z: U# S s7 f' ]
- NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); 0 Y% j$ |5 w( X& C) o- M
- My_USART1_Init();
& H1 ^) W0 n. P. `1 O - while(1);
& p/ A9 i* s+ V( f -
+ m- ?2 ?0 G* _( H% u - }
复制代码
6 n8 p3 O& |( x" f" s2 v区别就是,在UART实验中,USART1是和USB转串口模块连接在一起的,然后与电脑上的串口调试助手进行通信;现在改成USART1是和蓝牙模块连接在一起的,然后和手机上的蓝牙串口调试助手进行通信。
# T; R9 @5 b) i) ]
) }0 S4 s# N+ h& n2 e3 _6 D |