android 经典蓝牙编程 SPP
文章目录
前言
最近有一些蓝牙的通信需要做,就研究了一下蓝牙连接相关
- 连接蓝牙电子秤
- 连接 pos 机打印
其中连接蓝牙电子秤是接收数据 pos 机打印是发送数据/接收数据
流程图
流程图画的相当不专业,请自行脑补
核心类
因为这次的电子秤不是 4.0 的设备,所以没有使用 BLE 的开发,而是经典蓝牙(SPP)的连接方式 BluetoothAdapter
全局变量
1protected BluetoothAdapter mAdapter;
获取的方法,在 API18 下的时候使用的方式和以上的不一样,其实差别不大
1if (SDK_INT < 18) {
2 adapter = BluetoothAdapter.getDefaultAdapter();
3} else {
4 BluetoothManager bm = (BluetoothManager) getSystemService(BLUETOOTH_SERVICE);
5 adapter = bm.getAdapter();
6}
蓝牙使用的是注册广播的方式来获取系统给我们的通知 核心有以下的几种,注册广播的方法自己去搜下吧
1 BluetoothDevice.ACTION_FOUND//找到设备
2 BluetoothDevice.ACTION_NAME_CHANGED//设备的名字
3 BluetoothAdapter.ACTION_DISCOVERY_FINISHED//扫描结束
4 BluetoothDevice.ACTION_PAIRING_REQUEST//配对请求的放弃
5 BluetoothAdapter.STATE_OFF//蓝牙关闭
6 BluetoothAdapter.STATE_ON//蓝牙开启
连接
一般这种连接应该是全局单例,考虑写在了 service 中
首先需要扫描所有的蓝牙连接,但是这里有个坑,就是如果你将广播注册在 onCreate 中和 onDestroy,你每次都需要接收系统的广播,如果出现同名或者别的原因,这里就会一直接收广播 所以这里需要动态的将广播注册与反注册 我这里使用的方案请参照流程图
说明
这里是经典蓝牙(SPP)的连接代码,BLE 的连接相关目前还没有做,后续如果有实现的时候也会再写 blog
关于 UUID
蓝牙连接的 UUID 是有一套自己的定义规范的,但是目前的厂商不知道为什么大部分习惯都使用同一个 UUID,我这里蓝牙称,蓝牙打印机都用的这个
1 public static final UUID _UUID = java.util.UUID.fromString("00001101-0000-1000-8000-00805F9B34FB");
扫描
1 mAdapter.startDiscovery()
关于找到设备的坑
这里有一些关于设备的坑,有的设备你使用BluetoothDevice.ACTION_FOUND
广播中获取蓝牙设备的名字会发现是空的,这个时候就很尴尬了,使用系统自带的蓝牙扫描也会发现,扫描到的是先 MAC 地址,然后过几秒更改为名字,这里有另一个广播BluetoothDevice.ACTION_NAME_CHANGED
可以获取到名字更新时的光爆
当然蓝牙设备如果提前知道蓝牙的 mac 地址,最好还是使用 mac 地址连接匹配为最佳,毕竟 mac 地址轻易不会重复
配对
1 try {
2 // 连接建立之前的先配对
3 if (device.getBondState() == BluetoothDevice.BOND_NONE) {
4 Method creMethod = BluetoothDevice.class.getMethod("createBond");
5 creMethod.invoke(device);
6 } else {//已配对,连接socket
7 connDevice(device);
8 }
9 } catch (Exception e) {
10 // TODO: handle exception
11 //DisplayMessage("无法配对!");
12 e.printStackTrace();
13 }
这里如果已经配对,则直接连接,如果没有配对需要先配对,这里只是创建配对请求
ACTION_PAIRING_REQUEST
这个广播用于接收配对请求的发起,然后使用
1 LogUtils.d(TAG, "manualConn:" + manualConn);
2 if (manualConn) {
3 return;
4 }
5 final BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
6 if (device == null) {
7 return;
8 }
9 LogUtils.d(TAG, device.getName());
10 if (device.getBondState() != BluetoothDevice.BOND_BONDED) {
11 try {
12 // ClsUtils.setPin(device.getClass(), device, pwd); // 手机和蓝牙采集器配对
13 LogUtils.d(TAG, "设置pin码" + pwd);
14 ToastUtils.toast(device.getName() + "配对中");
15 boolean pinResult = device.setPin(pwd.getBytes());
16// handler.postDelayed(new Runnable() {
17// @Override
18// public void run() {
19// try {
20// ClsUtils.cancelPairingUserInput(device.getClass(), device);
21// } catch (Exception e) {
22// }
23// }
24// }, 1000);
25 // 一般调用不成功,前言里面讲解过了
26 LogUtils.d(TAG, "pinResult:" + pinResult);
27 if (pinResult) {
28 connDevice(device);
29 } else {
30 ToastUtils.toast("pin码不正确,请手动配对电子秤后点击连接");
31 manualConn = true;
32 }
33 } catch (Exception e) {
34 ToastUtils.toast("请求连接错误...");
35 }
36 }
这里是直接贴出来了程序中的片段,也会有一些报错,替换为自己的 Log/Toast 的 util 就可以了,也可以考虑删除掉,这里注释掉的部分原本是打算关闭配对的连接框的,但是这里有一些其他的问题,不同的 rom 处理方式不同,有些处理时会将 pin 码清空导致连接失败,所以这里将这部分代码清空,不关闭了
连接设备
核心连接代码
1socket = device.createInsecureRfcommSocketToServiceRecord(_UUID); //不安全的
2// socket = device.createRfcommSocketToServiceRecord(_UUID); //安全
3socket.connect()
关于连接方式,这里有两种.一种是使用不安全的连接方式,一种是使用安全的连接方式,这里需要根据你的蓝牙设备作为区分, 都是创建一个 BluetoothSocket
一般来说,如果对方是 android 设备,应该是安全的,如果是外置设备(蓝牙电子秤) 一般使用不安全的方案.
这两个连接方法要求 API>10
题外话
我其实现在的minSDK已经是19了,目前google自己都放弃维护低版本了,google chrome都不支持他们了,咱们也没必要继续支持低版本了
连接成功后,可以获取到一个 socket 连接,可以用这个 socket 连接获取到相关信息,不管是接收还是发出,通过这个连接就可以具体的获取相关信息
这里只是抛砖引玉
具体的实现后续有时间抽取出来形成一个开源项目再贴吧,项目毕竟是公司的,我这里不方便直接将整个代码贴出来,这个连接的相关代码非常乱
后记
这个蓝牙项目让我感受到了 google 深深的恶意,整个项目的坑数不胜数,这里先这样吧