Android ble (蓝牙低功耗)使用注意事项(转)

一、如何定义ble中service uuid?

  • 蓝牙标准规范里面定义了很多已经定义过的service uuid,如果冲突了会造成很多意外的问题。
  • 蓝牙的service uuid的格式如下

    UUID.fromString("00001234-0000-1000-8000-00805f9b34fb")

  • 在Android可以简单的采用这个原则:1、利用这个字符串【00002903-0000-1000-8000-00805f9b34fb】用第5-8位的数字做变化,其他数字保持不变。比如

    UUID.fromString("00007777-0000-1000-8000-00805f9b34fb")

    UUID.fromString("00009999-0000-1000-8000-00805f9b34fb")

二、ble中心设备开启扫描,设置所关心的serviceuuid。

bluetoothLeScanner = mBluetoothAdapter.getBluetoothLeScanner();
        List<ScanFilter> filters = new ArrayList<>();
       ScanFilter filter = new ScanFilter.Builder().setServiceUuid(ParcelUuid.fromString("00007777-0000-1000-8000-00805f9b34fb");)
                .build();
        filters.add(filter);
        ScanSettings scanSettings = new ScanSettings.Builder()
                .setScanMode(ScanSettings.SCAN_MODE_LOW_LATENCY)
                .build();
        bluetoothLeScanner.startScan(filters, scanSettings, scanCallback);

new ScanFilter.Builder().setServiceUuid(ParcelUuid.fromString("00007777-0000-1000-8000-00805f9b34fb");

三、ble外围设备可以在广播的时候设定AdvertiseData的magic number【manufacturerId 和 manufacturerSpecificData】。这样即使定义service uuid跟别人的有冲突,也可以在中心过滤该magic number来找到符合自己需求的外围设备

  • 外围构建AdvertiseData
AdvertiseData.Builder()
                .setIncludeDeviceName(true)
                .addServiceUuid(ParcelUuid.fromString("00007777-0000-1000-8000-00805f9b34fb"))
                .addManufacturerData(0x7777, new byte[]{0x07, 0x07})
                .build();
  • 中心处理AdvertiseData中的
final ScanCallback scanCallback = new ScanCallback() {
        @Override
        public void onScanResult(int callbackType, ScanResult result) {
            super.onScanResult(callbackType, result);
            ScanRecord scanRecord = result.getScanRecord();
            SparseArray<byte[]> mandufacturerData = scanRecord.getManufacturerSpecificData();

此时可以根据mandufacturerData来匹配自己设定的外围设备

四、什么时候ble的中心和外围才算真正的连接上?(可以开始传输数据了)

在BluetoothGattCallback中的关于此问题有三步回调

1、public void onConnectionStateChange(BluetoothGatt gatt, int status, int newState)

  • 这是ble中心和外围连接后,最先触发的回调。newstate等于BluetoothProfile.STATE_CONNECTED仅仅表示中心设备连接上了,这个时候需要去调用BluetoothGatt去发现服务。
  • 注意case1,在new state为BluetoothProfile.STATE_DISCONNECTED时,务必关掉BluetoothGatt,因为每次调用mBluetoothGatt = device.connectGatt(SpeakerApp.appContext, false, mGattCallBack);都会生成新的对象,而不会去主动关闭老的对象
  • 注意case2,133问题,iPhone 和 某些Android手机作为旁支会出现蓝牙初始连接就是133,此情况下应该立刻重新扫描连接。133问题链接
//iPhone 和 某些Android手机作为旁支会出现蓝牙初始连接就是133,此情况下立刻重试
            if (status == 133) {
                RLog.d(TAG, "发生设备初始连接133情况,需要重新扫描连接设备");
                mBluetoothGatt.close();
                //!!!需要去增加代码进行重新扫描重连
                return;
            }
            if (newState == BluetoothProfile.STATE_CONNECTED) {
                mBluetoothGatt = gatt;
                mBluetoothGatt.discoverServices();
            } else if (newState == BluetoothProfile.STATE_DISCONNECTED) {
                mBluetoothGatt.close();
                mBluetoothGatt = null;
            }

2、 public void onServicesDiscovered(BluetoothGatt gatt, int status)

mBluetoothGatt.discoverServices()执行后得到的callback,如果状态为GATT_SUCCESS,则可以获取ble旁支发起广播的service和descriptor,把广播设为enable

mCharacteristic = service.getCharacteristic(UUID.fromString("00007770-0000-1000-8000-00805f9b34fb"));
                if (mCharacteristic == null) {
                    RLog.e(TAG, "Can‘t find target characteristic.");
                    return;
                }
                gatt.setCharacteristicNotification(mCharacteristic, true);
                BluetoothGattDescriptor descriptor = mCharacteristic.getDescriptor(UUID.fromString("00007777-0000-1000-8000-00805f9b34fb"));
                descriptor.setValue(BluetoothGattDescriptor.ENABLE_NOTIFICATION_VALUE);
                gatt.writeDescriptor(descriptor);

3、public void onDescriptorWrite(BluetoothGatt gatt, BluetoothGattDescriptor descriptor, int status)

只有这一步status == BluetoothGatt.GATT_SUCCESS,才可以真正的传输数据,如果在第一步或者第二步就开始传输数据,会在某些特定的case下导致未知的bug或者空指针错误

所以,在中心设备跟外围开始连接后,你可以设定一个超时时间,在超时时间过后,依然没能回调onDescriptorWrite并获得BluetoothGatt.GATT_SUCCESS,则此次过程失败,你可以根据实际情况进行重连或者提示错误

五、mtu-20字节问题

mtu20的来源:GATT是基于ATT Protocol的,而它的 core spec里面定义了ATT的默认MTU为23个bytes,除去ATT的opcode一个字节以及ATT的handle2个字节之后,剩下的20个字节便是留给GATT的了

如果要传输大于20字节的数据怎么办?

1、 系统mtu可以支持修改到512字节,完成大数据量的传输。但是由于涉及到中心和旁支都需要修改,会造成很大的局限性和底层修改量,而且会触发比如某些设备第一次修改不生效,另一个设备一次连接中只能修改一次等bug,非常不可取,十分不建议。

2、分包传输,自己设计协议分包传输是最可取的方案,需要注意的是在分包后,每一个包之间写入数据需要设置间隔,比如100ms。

六、写数据之前做校验,判断获取的characteristic是否满足可读,可广播,或者需要回复等约定。

  return ((characteristic.getProperties() & BluetoothGattCharacteristic.PROPERTY_WRITE) > 0 ||
                (characteristic.getProperties() & BluetoothGattCharacteristic.PROPERTY_WRITE_NO_RESPONSE) > 0);

七、丢数据包问题

在做好5和6的基础上,依然会在一些设备上出现,由于系统原因,ble刚开始的发送第一个数据出现丢包,请对此做出特殊处理。

八、解析数据

  • 中心端mtu分包发给外围后,外围可以在

    public void onCharacteristicWriteRequest(BluetoothDevice device, int requestId, BluetoothGattCharacteristic characteristic, boolean preparedWrite, boolean responseNeeded, int offset, byte[] value) 接收到数据并还原成原始数据

  • 外围端mtu分包发给中心端后,中心端可以在 public void onCharacteristicChanged(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic) 接收到数据并还原成原始数据
  • 注意,对于一些蓝牙设备,总有一些特殊的状态,对于接受到的数据一定要进行正确性校验

九、other坑

  • ble中的PROPERTY_WRITE_NO_RESPONSE不可信任,google的有些版本并没有去读取这个属性值,而是直接设置为需要résponse,稳妥的方式最好设置为必须回复
  • 在项目中如果有多个ble或 ble + 经典蓝牙连接,在一些临界情况(比如设备重启,crash闪退重启),a ble连接可能需要移除b ble(或 b经典蓝牙)连接产生的设备,否则会导致a ble一直连接不上。
 BluetoothDevice remoteDevice = mBluetoothAdapter.getRemoteDevice("另一个ble 或者 蓝牙设备mac值");
                    if (remoteDevice.getBondState() == BluetoothDevice.BOND_BONDED) {
                        try {
                            Method removeBond = remoteDevice.getClass().getDeclaredMethod("removeBond");
                            removeBond.invoke(remoteDevice);
                            RLog.d(TAG , "成功移除系统bug");
                        } catch (Exception e) {
                            RLog.e(TAG , "反射异常");
                        }
                    }

作者:RDuwan.链接:

原文地址:https://www.cnblogs.com/sunupo/p/10405371.html

时间: 2024-11-06 16:56:15

Android ble (蓝牙低功耗)使用注意事项(转)的相关文章

Android BLE 蓝牙低功耗教程,中央BluetoothGatt和周边BluetoothGattServer的实现

Android4.3 规范了BLE的API,但是直到目前的4.4,还有些功能不完善. 在BLE协议中,有两个角色,周边(Periphery)和中央(Central):周边是数据提供者,中央是数据使用/处理者:在iOS SDK里面,可以把一个iOS设备作为一个周边,也可以作为一个中央:但是在Android SDK里面,直到目前最新的Android4.4.2,Android手机只能作为中央来使用和处理数据:那数据从哪儿来?从BLE设备来,现在的很多可穿戴设备都是用BLE来提供数据的. 一个中央可以同

Android BLE蓝牙详细解读

代码地址如下:http://www.demodashi.com/demo/15062.html 随着物联网时代的到来,越来越多的智能硬件设备开始流行起来,比如智能手环.心率检测仪.以及各式各样的智能家具和玩具类产品.安卓4.3(API 18)为BLE的核心功能提供平台支持和API,App可以利用它来发现设备.查询服务和读写特性.相比传统的蓝牙,BLE更显著的特点是低功耗.本文主要讲解Android低功耗蓝牙的api使用以及蓝牙扫描.连接.发送数据.接收数据等一系列操作,并主要介绍本人封装的Ble

Android ble 蓝牙4.0 总结一

本文介绍Android ble 蓝牙4.0,也就是说API level >= 18,且支持蓝牙4.0的手机才可以使用,如果手机系统版本API level < 18,也是用不了蓝牙4.0的哦. 首先发一下官方的demo,有兴趣的可以过去看看:http://developer.android.com/guide/topics/connectivity/bluetooth-le.html.android系统4.3以上,手机支持蓝牙4.0,具有搜索,配对,连接,发现服务及特征值,断开连接等功能,下载地

Android ble 蓝牙4.0 总结

本文介绍Android ble 蓝牙4.0,也就是说API level >= 18,且支持蓝牙4.0的手机才可以使用,如果手机系统版本API level < 18,也是用不了蓝牙4.0的哦. 首先发一下官方的demo,有兴趣的可以过去看看:http://developer.android.com/guide/topics/connectivity/bluetooth-le.html.android系统4.3以上,手机支持蓝牙4.0,具有搜索,配对,连接,发现服务及特征值,断开连接等功能,下载地

android ble蓝牙开发略解

Android 蓝牙4.0开发 1.  权限和相关属性 “android:required="true"表示apk只有在具有bluetooth_le属性的系统里运行,这个4.3之前android系统没有 <uses-featureandroid:name="android.hardware.bluetooth_le"android:required="true"/> <uses-permissionandroid:name=&q

Android ble蓝牙使用注意

以下均为自己在Android ble开发项目中遇到的问题 1.尽量不要在BluetoothGattCallback里面的回调函数中执行读写通知操作,最多一个,因为例如在onServicesDiscovered回调函数中只会传一个写操作,不管里面有多少个,而通知如setCharacteristicNotification(characterist,true)也有写操作,所以如果需要同时执行多步征特操作时,不能在回调函数中执行,不然只会执行第一步特征操作. 2.读写通知都是异步操作,但是一般必须一步

Android BLE (低功耗蓝牙)应用

蓝牙( Bluetooth? ):是一种无线技术标准,可实现固定设备.移动设备和楼宇个人域网之间的短距离数据交换(使用2.4-2.485GHz的ISM波段的UHF无线电波).蓝牙技术最初由电信巨头爱立信公司于1994年创制,当时是作为RS232数据线的替代方案.蓝牙可连接多个设备,克服了数据同步的难题. 如今蓝牙由蓝牙技术联盟(Bluetooth Special Interest Group,简称SIG)管理.蓝牙技术联盟在全球拥有超过25,000家成员公司,它们分布在电信.计算机.网络.和消费

Android BLE 蓝牙编程(二)

大家中秋快乐啊--哈哈,今天继续工程项目吧! 上篇我们已经实现了蓝牙设备的扫描,本篇我们来通过list展示扫描到的设备并 实现点击连接. 先贴出上篇的完整的MainActivity的方法: package com.wbnq.shouhuan; import android.bluetooth.BluetoothAdapter; import android.bluetooth.BluetoothDevice; import android.bluetooth.BluetoothManager;

Android BLE 蓝牙编程(三)

上节我们已经可以连接上蓝牙设备了. 本节我们就要获取手环的电池电量和计步啦. 在介绍这个之前我们需要先了解下什么是 服务 什么是 UUID 我们记得上节中我们item监听事件的回调的返回值是BluetoothGatt 类型的,还记得么?嘿嘿. 返回的bluetoothgatt中包含一个或多个BluetoothGattService(服务) 每个service包含一个或多个characteristic(特征值) 每个特征值包含一个value 和多个 descriptor(注意看啊!是一个value

Android BLE蓝牙开发-读写数据 获取UUID

https://www.jianshu.com/p/3711cfbf7128 一个ble蓝牙设备有多个包括多个Profile 一个Profile中有多个服务Service(通过服务的uuid找到对应的Service) 一个Service中有多个特征Characteristic(通过特征的uuid找到对应的Characteristic) 一个Characteristic中包括一个value和多个Descriptor(通过Descriptor的uuid找到对应的Descriptor) 另外注意,连接