登录

在这个站点登录

保存我的登录记录

<<忘记密码?

还没有账号?点此注册>>

Jerry

树莓派3B使用板载蓝牙与手机蓝牙进行Socket通信(RFCOMM)

分享到:

本文已被浏览19526

最近遇到一个项目,需要使用树莓派的板载蓝牙和手机进行通信。开发语言使用Python,对于Python直接调用蓝牙串口进行通信在Github上找到了一个pybluez的库(GitHub传送门),提供了基于套接字的蓝牙串口通信接口。

首先安装pybluez库,pybluez库会在使用pip安装时自动编译相关的蓝牙库,所以需要在安装pybluez之前先把蓝牙开发环境配置好:

sudo apt-get install Python-dev
sudo apt-get install libbluetooth-dev
sudo pip3 install pybluez

pybluez库的接口和Python中对于TCP的socket模块接口类似。
首先进行架构的设计。考虑到树莓派需要同时连接多个手机,接收多个手机发来的消息,并作出相应,所以应当将树莓派作为套接字的服务端,手机作为客户端。
先在手机上搞个串口调试工具,我直接在Google Play上找了一款名为蓝牙串口调试助手的App:

蓝牙串口调试助手 Pro

在开始的时候,发现树莓派的蓝牙有个问题:手机压根搜不到。这就尴尬了,如果搜不到树莓派,那么手机跟谁连接呢。后来在网上搜了一下,树莓派上有个程序叫bluetoothctl,可以进入一个蓝牙专用的命令行来操作蓝牙。蓝牙中需要开启发现才能让别的设备搜索到,进入bluetoothctl命令,使用discovery命令开启其他设备可见。

$ bluetoothctl
[bluetooth]# discoverable yes

开启树莓派蓝牙其他设备可见
当打开了可被其他设备发现后,手机上刷新列表后就能够看到树莓派了。
开启树莓派蓝牙其他设备可见

需要注意的是,树莓派的蓝牙可见和手机的类似,开启后有一定的延时,超过时间就会自动关闭可被其他设备发现。目前还没有找到方法能够保持可被发现的状态。

然后就是编写程序了。这里先介绍一下socket在服务器端的工作原理。作为一个服务器,需要监听端口,等待客户端的连接。当有客户端连接时,新开一个线程专门服务新链接,用于处理新连接的所有请求数据。

在pybluez中提供了一个BluetoothSocket类,用于生成socket对象。在BluetoothSocket类中拥有与socket类类似的结构,所以使用方法也类似。

第一步是创建一个BluetoothSocket对象,使其运行在服务器模式下,等待其他设备的连接,通信协议选择RFCOMM:

#创建一个服务器套接字,用来监听端口
server_socket=bluetooth.BluetoothSocket(bluetooth.RFCOMM);
#允许任何地址的主机连接,端口号
server_socket.bind(("",1))
#监听端口
server_socket.listen(1);

随后,使用accept函数接受新的连接,这个函数的返回值是元组,其中包含了新连接的信息,这里我使用的RFCOMM协议返回的元组中第一个是一个连接对象,另一个是一个信息表。这里直接使用一个死循环来处理这些事情,无需担心死循环会过多占用资源,对于accept函数,如果没有新链接,则会阻塞当前线程,也就不会占用CPU时间了。之后的就是开线程服务这个连接了。

#开死循环 等客户端连接
#本处应放在另外的子线程中
while True:
	#等待有人来连接,如果没人来,就阻塞线程等待
	sock,info=server_socket.accept();
	#打印有人来了的消息
	print(str(info[0])+' Connected!');
	#创建一个线程专门服务新来的连接
	t=threading.Thread(target=serveSocket,args=(sock,info[0]))
	#设置线程守护,防止程序在线程结束前结束
	t.setDaemon(True)
	#启动线程
	t.start();

在这个Demo中,设定为当客户端发来信息后,将其蓝牙地址与信息打印在控制台中并将其回传给客户端,与accept函数相同,在接收缓存中没有数据的时候recv函数也会将当前线程阻塞。

#连接套接字服务子线程
def serveSocket(sock,info):
	#开个死循环等客户端来信息
	while True:
		#接收1024个字节,然后以UTF-8解码(中文),如果没有可以接收的信息则自动阻塞线程(API)
		receive=sock.recv(1024).decode('utf-8');
		#打印刚刚读到的东西(info=地址)
		print('['+str(info)+']'+receive);
		#为了返回好看点,加个换行
		receive=receive+"\n";
		#回传数据给发送者
		sock.send(receive.encode('utf-8'));

于是就得到了如下的运行效果
树莓派控制台
手机端
综上所述,当树莓派作为服务器端时的程序:

#-*- coding:utf-8 -*-
import bluetooth
import threading
#服务器套接字(用来接收新链接)
server_socket=None

#连接套接字服务子线程
def serveSocket(sock,info):
	#开个死循环等待客户端发送信息
	while True:
		#接收1024个字节,然后以UTF-8解码(中文),如果没有可以接收的信息则自动阻塞线程(API)
		receive=sock.recv(1024).decode('utf-8');
		#打印刚刚读到的东西(info=地址)
		print('['+str(info)+']'+receive);
		#为了返回好看点,加个换行
		receive=receive+"\n";
		#回传数据给发送者
		sock.send(receive.encode('utf-8'));

#主线程

#创建一个服务器套接字,用来监听端口
server_socket=bluetooth.BluetoothSocket(bluetooth.RFCOMM);
#允许任何地址的主机连接,未知参数:1(端口号,通道号)
server_socket.bind(("",1))
#监听端口/通道
server_socket.listen(1);

#开死循环 等待客户端连接
#本处应放在另外的子线程中
while True:
	#等待有人来连接,如果没人来,就阻塞线程等待(这本来要搞个会话池,以方便给不同的设备发送数据)
	sock,info=server_socket.accept();
	#打印有人来了的消息
	print(str(info[0])+' Connected!');
	#创建一个线程专门服务新来的连接(这本来应该搞个线程池来管理线程的)
	t=threading.Thread(target=serveSocket,args=(sock,info[0]))
	#设置线程守护,防止程序在线程结束前结束
	t.setDaemon(True)
	#启动线程
	t.start();

另外,树莓派也可以作为客户端访问其他设备也可以通过BluetoothSocket类,只不过在调用bind部分换成了调用connect函数。

 手机扫描左边的二维码,立刻将文章收入手机!
 微信扫描左边二维码,点击右上角即可分享到朋友圈!
严禁任何非授权的采集与转载,转载须经站长同意并在文章显著位置标注本文连接,站长保留追究法律责任的权利.

评论

 您需要 先登录 才可以回复.