Android开发——蓝牙多设备连接(四)

本文详细介绍了在Android开发中实现蓝牙多设备连接的场景分析、效果展示及其实现过程,包括多设备连接、设备自动连接、数据共享、蓝牙搜索连接删除等关键点。同时探讨了在开发过程中遇到的问题和解决方案,以及对未来数据存储优化的思考。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

经过一个多月的时间蓝牙多设备连接的重构终于告一段落了,这次的重构不止是代码方面的完善,还结合了一些用户的使用场景,另外增加一些离线操作,使手机端对蓝牙的操作更加的便捷,对蓝牙设备的管理更加统一。

场景分析

支持的场景
  • 多设备连接(一个血压计和一个厨房秤)—— 在饭点,使用同一个手机连接血压计和厨房秤,病人在测量血压,家人在做饭称量菜品。
  • 把连接到的设备信息保存到本地——方便下次自动连接。
  • 把蓝牙返回结果保存本地——方便在其他页面获取数据显示UI。
  • 同种设备只能连接一个——保持连接设备的唯一性,防止找不到连接的设备。
  • 可以在蓝牙列表扫描到所支持的设备,并连接、断开和删除蓝牙——对蓝牙设备进行管理,查看自己扫描到的设备,并查看蓝牙的连接状态。
  • 启动 App 可以自动连接上一次已经连接过的设备——可以快速的使用设备。
暂不支持的场景
  • 同时扫描多个同种设备——可能导致连接到了设备,而不知道连接的是哪个设备。
  • 新旧设备替换——不能直接进行替换,需要在设备列表页面删除旧设备,之后关掉旧设备,重新扫描新设备。

场景实现

效果展示

蓝牙多设备连接效果图.gif

多设备连接

现在一台手机可以连接多个设备,例如连接蓝牙耳机,智能手环等。既然手机可以连接多个设备,那么移动应用也是可以连接多个设备的(血压计、心率计等),下面就是移动应用 App 实现多设备连接的思路方法。

实现思路

关于蓝牙连接,主要是 BluetoothGatt 这个类型,每个蓝牙的连接都需要用独立且唯一的 BluetoothGatt 。开始的想法是每个蓝牙都重新创建一个 Service, 在新的 Service 内使用 BluetoothGatt 进行连接,然而这个方法是可以实现多设备连接,但是创建多个 Service 对手机消耗比较大。之后,想到把 BluetoothGatt 保存起来不就可以了么,那用什么保存呢,既可以临时保存多个,又可以按照需要获取相对应的 BluetoothGatt 。在 java 里面有个类型 Map(String, Object) ,它是以 key-value 的形式存储到 Map 中。可以根据当时的 Key 来取相应的 Value 值,而且在关掉进程时相应的变量也就释放了。

代码实现
    private Map<String, BluetoothGatt> mBluetoothGattMap = new HashMap<>(); //临时保存 BluetoothGatt
    private Map<String, BluetoothGattCharacteristic> mGattCharacteristicMap = new HashMap<>();// 临时保存蓝牙的特征值 Characteristic
    private Map<String, BluetoothInfo> mBluetoothInfoMap = new HashMap<>();// 临时保存自己设置的蓝牙信息(deviceName、deviceType、startCMD、stopCMD 等)
    private Map<BluetoothGatt, String> mDeviceTypeMap = new HashMap<>();// 临时存储 deviceType 
    private Map<String, GGBLEDeviceEntity> mConnectModelMap = new HashMap<>();// 临时存储 已连接的设备
//...
/**
     * 连接设备
     *
     * @param deviceType 设备类型
     * @return true 连接成功,false连接失败
     */
    boolean connectBluetooth(Context context, String deviceType, String deviceAddress) {
   
   
       //...
        BluetoothDevice device = mBluetoothAdapter.getRemoteDevice(deviceAddress);//根据 mac 地址获取蓝牙设备
      //...
        BluetoothGatt bluetoothGatt = device.connectGatt(this, false, mGattCallback);// 对蓝牙进行连接操作
      //...
        mBluetoothGattMap.put(deviceType, bluetoothGatt);//把 BluetoothGatt 已 key-value 的形式临时保存起来

        return true;

    }

设备自动连接

要有良好的用户操作体验,我们应该避免对一些无关的操作重复进行。例如我们第一次打开应用连接了蓝牙设备,以后再打开 App 不需要重复操作连接过程,用户就可以少打开一个页面,少点击两次按钮。减少用户重复操作,让用户直接进入正题,提高主功能的使用率。

实现思路

在 Android 中连接蓝牙的方法是

public BluetoothGatt connectGatt
### Android BluetoothGatt 使用方法 #### 1. 初始化蓝牙适配器并检查 BLE 支持 为了使用 `BluetoothGatt` 功能,首先需要初始化蓝牙适配器,并确认设备是否支持 BLE 功能。这一步骤通常涉及读取系统的蓝牙状态以及验证必要的权限。 在应用程序的清单文件 (`AndroidManifest.xml`) 中需声明以下权限和硬件特性[^4]: ```xml <uses-permission android:name="android.permission.BLUETOOTH"/> <uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/> <uses-feature android:name="android.hardware.bluetooth_le" android:required="true"/> ``` 如果目标 API 级别为 23 或更高,则还需要动态请求位置权限,因为从 Android 6.0 开始,扫描附近设备需要访问粗略或精确的位置数据。 #### 2. 连接到远程设备 通过调用 `connectToDevice` 方法可以建立与远程设备的连接。此操作返回一个 `BluetoothGatt` 实例,该实例用于管理后续的所有 GATT 客户端交互。以下是典型的连接代码片段: ```java bluetoothGatt = device.connectGatt( context, false, // 不启用自动重连 gattCallback, // 自定义回调对象 BluetoothDevice.TRANSPORT_LE); // 指定低功耗传输模式 ``` 上述代码中的参数解释已在引用中提供[^3]。 #### 3. 处理 GATT 回调事件 当成功连接到远程设备后,所有的通信都依赖于 `BluetoothGattCallback` 类型的对象。开发者可以在其中监听各种事件,例如服务发现完成、特征值更新等。 下面是一个简单的回调实现示例: ```java private final BluetoothGattCallback gattCallback = new BluetoothGattCallback() { @Override public void onConnectionStateChange(BluetoothGatt gatt, int status, int newState) { super.onConnectionStateChange(gatt, status, newState); if (newState == BluetoothProfile.STATE_CONNECTED && status == BluetoothGatt.GATT_SUCCESS){ Log.d(TAG,"Connected to the remote device."); gatt.discoverServices(); // 发现服务 }else{ Log.e(TAG,"Disconnected from the remote device or connection failed with error code:"+status); } } @Override public void onServicesDiscovered(BluetoothGatt gatt, int status) { super.onServicesDiscovered(gatt, status); if(status==BluetoothGatt.GATT_SUCCESS){ List<BluetoothGattService> services=gatt.getServices(); for(BluetoothGattService service :services){ Log.i(TAG,"Found Service UUID:" +service.getUuid().toString()); } } } }; ``` #### 常见问题分析 ##### A. 设备无法找到 BLE 特征值 可能的原因包括未正确执行服务发现流程或者目标设备的服务配置不符合预期。确保在接收到 `onServicesDiscovered` 回调后再尝试获取任何特定的 `BluetoothGattCharacteristic` 对象[^2]。 ##### B. 数据写入失败 某些情况下即使发送命令给外设也可能得不到响应。这是由于没有设置正确的属性标志位所致。比如要向某个特征值写入数据之前应该先判断其是否允许写操作(`PROPERTY_WRITE_NO_RESPONSE` 或者 `PROPERTY_WRITE`)。 ##### C. 断开连接异常频繁 断链现象可能是由多种因素引起的,如信号强度弱、干扰源存在或是程序逻辑错误造成的资源泄漏等问题。建议增加心跳包机制保持链接稳定性;同时注意释放不再使用的 GATT 资源以减少内存占用率。 --- ###
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值