一、蓝牙简介
① 概念
- 蓝牙是短距离无线通信的一种方式,支持蓝牙的两个设备必须配对后才能通信。HarmonyOS 蓝牙主要分为传统蓝牙和低功耗蓝牙(通常称为 BLE,Bluetooth Low Energy)。
- 传统蓝牙指的是蓝牙版本 3.0 以下的蓝牙,低功耗蓝牙指的是蓝牙版本 4.0 以上的蓝牙。
- 当前蓝牙的配对方式有两种:蓝牙协议 2.0 以下支持 PIN 码(Personal Identification Number,个人识别码)配对,蓝牙协议 2.1 以上支持简单配对。
② 传统蓝牙
- 传统蓝牙本机管理:打开和关闭蓝牙、设置和获取本机蓝牙名称、扫描和取消扫描周边蓝牙设备、获取本机蓝牙 profile 对其他设备的连接状态、获取本机蓝牙已配对的蓝牙设备列表。
- 传统蓝牙远端设备操作:查询远端蓝牙设备名称和 MAC 地址、设备类型和配对状态,以及向远端蓝牙设备发起配对。
③ BLE
- BLE 设备交互时会分为不同的角色:
-
- 中心设备和外围设备:中心设备负责扫描外围设备、发现广播。外围设备负责发送广播。
-
- GATT(Generic Attribute Profile,通用属性配置文件)服务端与 GATT 客户端:两台设备建立连接后,其中一台作为 GATT 服务端,另一台作为 GATT 客户端。
- HarmonyOS 低功耗蓝牙提供的功能有:
-
- BLE 扫描和广播:根据指定状态获取外围设备、启动或停止 BLE 扫描、广播。
-
- BLE 中心设备与外围设备进行数据交互:BLE 外围设备和中心设备建立 GATT 连接后,中心设备可以查询外围设备支持的各种数据,向外围设备发起数据请求,并向其写入特征值数据。
-
- BLE 外围设备数据管理:BLE 外围设备作为服务端,可以接收来自中心设备(客户端)的 GATT 连接请求,应答来自中心设备的特征值内容读取和写入请求,并向中心设备提供数据。同时外围设备还可以主动向中心设备发送数据。
④ 权限申请
- 调用蓝牙的打开接口需要 ohos.permission.USE_BLUETOOTH 权限。
- 调用蓝牙扫描接口需要 ohos.permission.LOCATION 权限和 ohos.permission.DISCOVER_BLUETOOTH 权限。
二、传统蓝牙本机管理
① 应用场景
- 传统蓝牙本机管理主要是针对蓝牙本机的基本操作,包括打开和关闭蓝牙、设置和获取本机蓝牙名称、扫描和取消扫描周边蓝牙设备、获取本机蓝牙profile对其他设备的连接状态、获取本机蓝牙已配对的蓝牙设备列表。
② API 说明
- 蓝牙本机管理类 BluetoothHost 的主要接口:
接口名 |
功能描述 |
getDefaultHost(Context context) |
获取BluetoothHost实例,去管理本机蓝牙操作 |
enableBt() |
打开本机蓝牙 |
disableBt() |
关闭本机蓝牙 |
setLocalName(String name) |
设置本机蓝牙名称 |
getLocalName() |
获取本机蓝牙名称 |
getBtState() |
获取本机蓝牙状态 |
startBtDiscovery() |
发起蓝牙设备扫描 |
cancelBtDiscovery() |
取消蓝牙设备扫描 |
isBtDiscovering() |
检查蓝牙是否在扫描设备中 |
getProfileConnState(int profile) |
获取本机蓝牙profile对其他设备的连接状态 |
getPairedDevices() |
获取本机蓝牙已配对的蓝牙设备列 |
③ 打开蓝牙
- 调用 BluetoothHost 的 getDefaultHost(Context context) 接口,获取 BluetoothHost 实例,管理本机蓝牙操作。
- 调用 enableBt() 接口,打开蓝牙。
- 调用 getBtState(),查询蓝牙是否打开。
// 获取蓝牙本机管理对象
BluetoothHost bluetoothHost = BluetoothHost.getDefaultHost(context);
// 调用打开接口
bluetoothHost.enableBt();
// 调用获取蓝牙开关状态接口
int state = bluetoothHost.getBtState();
④ 蓝牙扫描
- 开始蓝牙扫描前要先注册广播 BluetoothRemoteDevice.EVENT_DEVICE_DISCOVERED。
- 调用 startBtDiscovery() 接口开始进行扫描外围设备。
- 如果想要获取扫描到的设备,必须在注册广播时继承实现 CommonEventSubscriber 类的 onReceiveEvent(CommonEventData data) 方法,并接收 EVENT_DEVICE_DISCOVERED 广播。
// 开始扫描
bluetoothHost.startBtDiscovery();
// 接收系统广播
public class MyCommonEventSubscriber extends CommonEventSubscriber {
@Override
public void onReceiveEvent(CommonEventData data) {
if (data == null) {
return;
}
Intent info = data.getIntent();
if (info == null) {
return;
}
// 获取系统广播的action
String action = info.getAction();
// 判断是否为扫描到设备的广播
if (BluetoothRemoteDevice.EVENT_DEVICE_DISCOVERED.equals(action)) {
IntentParams myParam = info.getParams();
BluetoothRemoteDevice device = (BluetoothRemoteDevice) myParam.getParam(BluetoothRemoteDevice.REMOTE_DEVICE_PARAM_DEVICE);
}
}
}
三、传统蓝牙远端设备操作
① 应用场景
- 传统蓝牙远端管理操作主要是针对远端蓝牙设备的基本操作,包括获取远端蓝牙设备地址、类型、名称和配对状态,以及向远端设备发起配对。
② API 说明
- 蓝牙远端设备管理类 BluetoothRemoteDevice 的主要接口:
接口名 |
功能描述 |
getDeviceAddr() |
获取远端蓝牙设备地址 |
getDeviceClass() |
获取远端蓝牙设备类型 |
getDeviceName() |
获取远端蓝牙设备名称 |
getPairState() |
获取远端设备配对状态 |
startPair() |
向远端设备发起配对 |
③ 开发流程
- 调用 BluetoothHost 的 getDefaultHost(Context context) 接口,获取 BluetoothHost 实例,管理本机蓝牙操作。
- 调用 enableBt() 接口,打开蓝牙。
- 调用 startBtDiscovery(),扫描设备。
- 调用 startPair(),发起配对。
// 获取蓝牙本机管理对象
BluetoothHost bluetoothHost = BluetoothHost.getDefaultHost(context);
// 调用打开接口
bluetoothHost.enableBt();
// 调用扫描接口
bluetoothHost.startBtDiscovery();
//设置界面会显示出扫描结果列表,点击蓝牙设备去配对
BluetoothRemoteDevice device = bluetoothHost.getRemoteDev(TEST_ADDRESS);
device.startPair();
四、BLE 扫描和广播
① 应用场景
- 通过 BLE 扫描和广播提供的开放能力,可以根据指定状态获取外围设备、启动或停止 BLE 扫描、广播。
② API 说明
- BLE 中心设备管理类 BleCentralManager 的主要接口:
接口名 |
功能描述 |
startScan(List filters) |
进行BLE蓝牙扫描,并使用filters对结果进行过滤 |
stopScan() |
停止BLE蓝牙扫描 |
getDevicesByStates(int[] states) |
根据状态获取连接的外围设备 |
BleCentralManager(Context context, BleCentralManagerCallback callback) |
获取中心设备管理对象 |
- 中心设备管理回调类 BleCentralManagerCallback 的主要接口:
接口名 |
功能描述 |
scanResultEvent(BleScanResult result) |
扫描到BLE设备的结果回调 |
groupScanResultsEvent(List scanResults) |
扫描到一组BLE设备的结果回调 |
scanFailedEvent(int resultCode) |
启动扫描失败的回调 |
- BLE 广播相关的 BleAdvertiser 类和 BleAdvertiseCallback 类的主要接口:
接口名 |
功能描述 |
BleAdvertiser(Context context, BleAdvertiseCallback callback) |
用于获取广播操作对象 |
startAdvertising(BleAdvertiseSettings settings, BleAdvertiseData advData, BleAdvertiseData scanResponse) |
进行BLE广播,第一个参数为广播参数,第二个为广播数据,第三个参数是扫描和广播数据参数的响应 |
stopAdvertising() |
停止BLE广播 |
startResultEvent(int result) |
广播回调结果 |
③ 中心设备进行 BLE 扫描
- 进行 BLE 扫描之前先要继承 BleCentralManagerCallback 类实现 scanResultEvent 和 scanFailedEvent 回调函数,用于接收扫描结果。
- 调用 BleCentralManager(BleCentralManagerCallback callback) 接口获取中心设备管理对象。
- 获取扫描过滤器,过滤器为空时为不使用过滤器扫描,然后调用 startScan() 开始扫描 BLE 设备,在回调中获取扫描到的 BLE 设备。
// 实现扫描回调
public class ScanCallback implements BleCentralManagerCallback {
List<BleScanResult> results = new ArrayList<BleScanResult>();
@Override
public void scanResultEvent(BleScanResult resultCode) {
// 对扫描结果进行处理
results.add(resultCode);
}
@Override
public void scanFailedEvent(int resultCode) {
HiLog.warn(TAG,"Start Scan failed, Code: %{public}d", resultCode);
}
@Override
public void groupScanResultsEvent(final List<BleScanResult> scanResults){
// 对扫描结果进行处理
}
}
// 获取中心设备管理对象
private ScanCallback centralManagerCallback = new ScanCallback();
private BleCentralManager centralManager = new BleCentralManager(context, centralManagerCallback);
// 创建扫描过滤器然后开始扫描
List<BleScanFilter> filters = new ArrayList<BleScanFilter>();
centralManager.startScan(filters);
④ 外围设备进行 BLE 广播
- 进行 BLE 广播前需要先继承 advertiseCallback 类实现 startResultEvent 回调,用于获取广播结果。
- 调用接口 BleAdvertiser(Context context, BleAdvertiseCallback callback) 获取广播对象,构造广播参数和广播数据。
- 调用 startAdvertising(BleAdvertiseSettings settings, BleAdvertiseData advData, BleAdvertiseData scanResponse) 接口开始 BLE 广播。
// 实现BLE广播回调
private BleAdvertiseCallback advertiseCallback = new BleAdvertiseCallback() {
@Override
public void startResultEvent(int result) {
if(result == BleAdvertiseCallback.RESULT_SUCC){
// 开始BLE广播成功
}else {
// 开始BLE广播失败
}
}
};
// 获取BLE广播对象
private BleAdvertiser advertiser = new BleAdvertiser(this,advertiseCallback);
// 创建BLE广播参数和数据
private BleAdvertiseData data = new BleAdvertiseData.Builder()
.addServiceUuid(SequenceUuid.uuidFromString(Server_UUID)) // 添加服务的UUID
.addServiceData(SequenceUuid.uuidFromString(Server_UUID), new byte[]{0x11}) // 添加广播数据内容
.build();
private BleAdvertiseSettings advertiseSettings = new BleAdvertiseSettings.Builder()
.setConnectable(true) // 设置是否可连接广播
.setInterval(BleAdvertiseSettings.INTERVAL_SLOT_DEFAULT) // 设置广播间隔
.setTxPower(BleAdvertiseSettings.TX_POWER_DEFAULT) // 设置广播功率
.build();
// 开始广播
advertiser.startAdvertising(advertiseSettings, data, null);
五、BLE 中心设备与外围设备进行数据交互
① 应用场景
- BLE 外围设备和中心设备建立 GATT 连接,通过该连接中心设备可以获取外围设备支持的 Service、Characteristic、Descriptor、RSSI 等数据。同时,中心设备可以向外围设备进行数据请求,并向外围设备写入 Characteristic、Descriptor 等特征值数据。
② API 说明
- 低功耗蓝牙外围设备操作类 BlePeripheralDevice 的接口说明如下:
接口名 |
功能描述 |
connect(boolean isAutoConnect, BlePeripheraCallback callback) |
重新连接GATT外围设备,isAutoConnect表示是否自动进行连接 |
discoverServices() |
搜索外围设备支持的服务,特征和描述 |
getServices() |
获取外围设备支持的所有GATT服务 |
getService(UUID uuid) |
根据UUID获取外围设备支持的某个GATT服务 |
disconnect() |
与外围设备断开BLE连接 |
close() |
关闭蓝牙GATT客户端 |
readCharacteristic(GattCharacteristic characteristic) |
读取外围设备GATT特征 |
writeCharacteristic(GattCharacteristic characteristic) |
写指定外围设备的GATT特征值 |
setNotifyCharacteristic(GattCharacteristic characteristic, boolean enable) |
设置指定GATT特征通知的使能/去使能 |
readDescriptor (GattDescriptor descriptor) |
读取外围设备GATT描述值 |
writeDescriptor(GattDescriptor descriptor) |
写指定外围设备的GATT描述值 |
readRemoteRssiValue() |
读取已连接外围设备的RSSI |
requestBleConnectionPriority(int connPriority) |
请求链接参数更新 |
requestBleMtuSize(int mtu) |
请求用于给定连接的MTU大小 |
- 低功耗蓝牙外围设备操作回调类 BlePeripheralCallback 的接口说明如下:
接口名 |
功能描述 |
servicesDiscoveredEvent(int status) |
外围设备服务发生更新触发的回调 |
connectionStateChangedEvent(int connectionState) |
外围设备GATT连接状态发生变化时的回调 |
characteristicReadEvent(GattCharacteristic characteristic, int ret) |
GATT特征值读操作回调 |
characteristicWriteEvent(GattCharacteristic characteristic, int ret) |
GATT特征值写操作回调 |
characteristicChangedEvent(GattCharacteristic characteristic) |
外围设备特征通知触发的回调 |
descriptorReadEvent(GattDescriptor descriptor, int ret) |
GATT描述值读操作回调 |
descriptorWriteEvent(GattDescriptor descriptor, int ret) |
GATT描述值写操作回调 |
readRemoteRssiEvent(int rssi, int ret) |
外围设备发来读取RSSI的回调 |
mtuUpdateEvent(int mtu, int ret) |
GATT设备链接的MTU变化通知的回调 |
③ 开发流程
- 调用 startScan() 接口启动 BLE 扫描来获取外围设备。
- 获取到外围设备后,调用 connect(boolean isAutoConnect, BlePeripheraCallback callback) 建立与外围 BLE 设备的 GATT 连接,boolean 参数 isAutoConnect 用于设置是否允许设备在可发现距离内自动建立 GATT 连接。
- 启动 GATT 连接后,会触发 connectionStateChangedEvent(int connectionState) 回调,根据回调结果判断是否连接 GATT 成功。
- 在 GATT 连接成功时,中心设备可以调用 discoverServices() 接口,获取外围设备支持的 Services、Characteristics 等特征值,在回调 servicesDiscoveredEvent(int status) 中获取外围设备支持的服务和特征值,并根据 UUID 判断是什么服务。
- 根据获取到的服务和特征值,调用 read 和 write 方法可以读取或者写入对应特征值数据。
private List<GattService> services;
// 创建扫描过滤器然后开始扫描
List<BleScanFilter> filters = new ArrayList<BleScanFilter>();
centralManager.startScan(filters);
// 与外围设备建立连接,允许自动回连,连接会触发connectionStateChangedEvent回调
MyBlePeripheralCallback callback = new MyBlePeripheralCallback();
peripheraDevice.connect(true, callback);
// 连接后读取外围设备RSSI值,获取后触发readRemoteRssiEvent()回调
peripheraDevice.readRemoteRssiValue();
// 实现外围设备操作回调
private class MyBlePeripheralCallback extends BlePeripheralCallback {
@Override
public void connectionStateChangedEvent(int connectionState) { // GATT连接状态变化回调
if (connectionState == ProfileBase.STATE_CONNECTED){
peripheraDevice.discoverServices(); // 连接成功获取外围设备的Service
}
}
@Override
public void servicesDiscoveredEvent(int status) { // 获取外围设备Service的回调
if (status == BlePeripheralDevice.OPERATION_SUCC){
services = peripheraDevice.getServices(); // 获取Service成功后获服务列表
for (GattService service : services){
// 对每个服务进行相应操作
}
}
}
@Override
public void characteristicChangedEvent(GattCharacteristic charecteristic) { // 外围设备主动向中心设备发送特征值通知时触发回调
// 根据通知的charecteristic获取特征值携带的数据
}
@Override
public void characteristicWriteEvent(GattCharacteristic charecteristic, int ret) {
if (ret == BlePeripheralDevice.OPERATION_SUCC){
// 向外围设备写特征值数据成功后的操作
}
}
@Override
public void characteristicReadEvent(GattCharacteristic charecteristic, int ret) {
if (ret == BlePeripheralDevice.OPERATION_SUCC){
// 向外围设备写特征值数据成功后的操作
}
}
@Override
public void descriptorReadEvent(GattDescriptor descriptor, int ret) {
// 向外围设备读描述值数据成功后的操作
}
@Override
public void descriptorWriteEvent(GattDescriptor descriptor, int ret) {
// 向外围设备写描述值数据成功后的操作
}
@Override
public void readRemoteRssiEvent(int rssi, int ret) {
if (ret == BlePeripheralDevice.OPERATION_SUCC){
// 读取外围设备RSSI值成功后的操作,对端RSSI值为rssi
}
}
}
六、BLE 外围设备数据管理
① 应用场景
- BLE 外围设备作为服务端,可以接收来自中心设备(客户端)的 GATT 连接请求,应答来自中心设备的特征值内容读取和写入请求,并向中心设备提供数据,从而实现信息交互和消息同步。同时外围设备还可以主动向中心设备发送数据。
② API 说明
- 低功耗蓝牙外围设备操作类 BlePeripheralManager 的接口说明如下:
接口名 |
功能描述 |
BlePeripheralManager(Context context, BlePeripheralManagerCallback callback, int transport) |
获取外围设备管理回调对象 |
getServices() |
获取外围设备的所有服务 |
addService(GattService service) |
将GATT服务加入服务端 |
cancelConnection(BlePeripheralDevice device) |
取消与中心设备的GATT连接 |
clearServices() |
删除所有的GATT服务 |
close() |
关闭GATT服务端 |
getDevicesByStates(int[] states) |
通过状态获取连接的中心设备列表 |
notifyCharacteristicChanged(BlePeripheralDevice device, GattCharacteristic characteristic, boolean confirm) |
通知中心设备特征值出现变化 |
removeGattService(GattService service) |
移除特定的服务 |
sendResponse(BlePeripheralDevice device, int requestId, int status, int offset, byte[] value) |
发送回复给中心设备 |
- 低功耗蓝牙外围设备管理回调类 BlePeripheralManagerCallback 的接口说明如下:
接口名 |
功能描述 |
receiveCharacteristicReadEvent(BlePeripheralDevice device, int requestId, int offset, GattCharacteristic characteristic) |
收到中心设备对特征值的读取请求回调 |
receiveCharacteristicWriteEvent(BlePeripheralDevice device, int requestId, GattCharacteristic characteristic, boolean isPrep, boolean needRsp, int offset, byte[] value) |
收到中心设备对特征值的写入请求 |
connectionStateChangeEvent(BlePeripheralDevice device, int interval, int latency, int timeout, int status) |
中心设备连接事件回调 |
receiveDescriptorReadEvent(BlePeripheralDevice device, int transId, int offset, GattDescriptor descriptor) |
收到中心设备对描述值的读取请求回调 |
receiveDescriptorWriteRequestEvent(BlePeripheralDevice device, int transId, GattDescriptor descriptor, boolean isPrep, boolean needRsp, int offset, byte[] value) |
收到中心设备对描述值的写入请求回调 |
executeWriteEvent(BlePeripheralDevice device, int requestId, boolean execute) |
向中心设备执行写入操作的回调 |
mtuUpdateEvent(BlePeripheralDevice device, int mtu) |
中心设备MTU值变化的回调 |
notificationSentEvent(BlePeripheralDevice device, int status) |
向中心设备发送通知的回调 |
serviceAddedEvent(int status, GattService service) |
向外围设备添加服务结果回调 |
③ 开发流程
- 调用 BlePeripheralManager(Context context, BlePeripheralManagerCallback callback, int transport) 接口创建外围设备服务端并开启服务。
- 调用 GattService(UUID uuid, boolean isPrimary) 接口创建服务对象,向外围设备添加服务。
- 从回调接口 onCharacteristicWriteRequest 中获取中心设备发送来的消息,调用 notifyCharacteristicChanged 接口向中心设备发送通知。
private BlePeripheralDevice blePeripheralDevice;
private MyBlePeripheralManagerCallback peripheralManagerCallback = new MyBlePeripheralManagerCallback();
// 创建外围设备服务端并开启服务
private BlePeripheralManager peripheralManager = new BlePeripheralManager(this, peripheralManagerCallback, 0);
// 创建服务对象,向外围设备添加服务
private GattService service = new GattService(UUID.fromString(Service_UUID), true);
private GattCharacteristic writeCharacteristic = new GattCharacteristic(Character_UUID,
GattCharacteristic.PROPERTY_WRITE, GattCharacteristic.PROPERTY_WRITE);
service.addCharacteristic(writeCharacteristic);
peripheralManager.addServerService(service);
// 向中心设备发送通知
writeCharacteristic.setValue(data);
peripheralManager.notifyCharacteristicChanged(blePeripheralDevice, writeCharacteristic, false);
// 外围设备管理回调
public class MyBlePeripheralManagerCallback extends BlePeripheralManagerCallback {
// 中心设备向外围设备写入数据回调
@Override
public void receiveCharacteristicWriteEvent(BlePeripheralDevice device, int requestId,
GattCharacteristic characteristic, boolean isPrep, boolean needRsp, int offset, byte[] value){
// 中心设备写入外围设备的数据为value,对数据进行处理
}
// 外围设备连接状态变化回调
@Override
public void connectionStateChangeEvent(BlePeripheralDevice device, int interval, int latency, int timeout, int status){
if (status == ProfileBase.STATE_CONNECTED) {
// 中心设备连接服务端成功
blePeripheralDevice = device;
}
}
// 向中心设备发送通知回调
@Override
public void notificationSentEvent(BlePeripheralDevice device, int status){
if (status == BlePeripheralDevice.OPERATION_SUCC) {
// 向对中心设备发送通知成功回调
}
}
}