0

0

Android BLE AdvertisingSet 广播扫描响应数据配置指南

心靈之曲

心靈之曲

发布时间:2025-08-14 23:06:11

|

765人浏览过

|

来源于php中文网

原创

android ble advertisingset 广播扫描响应数据配置指南

本文旨在解决Android BLE AdvertisingSet在广播时无法正确发送扫描响应数据的问题。核心在于配置AdvertisingSetParameters时,必须显式调用setScannable(true),以允许设备响应扫描请求并发送包含额外数据的扫描响应包。教程将详细阐述AdvertisingSet的配置、扫描响应的发送机制以及相关的代码实现与注意事项,确保BLE广播数据能够完整被接收。

1. 理解 BLE 广播与扫描响应

蓝牙低功耗(BLE)设备通过广播(Advertising)来对外发布自身的存在和能力。广播数据通常包含设备名称、服务UUID等基本信息。然而,广播数据包的大小是有限的(在传统广播模式下为31字节)。当需要传输更多数据时,BLE引入了“扫描响应”(Scan Response)机制。

当一个扫描器(Scanner)检测到某个广播设备并需要获取更多信息时,它可以发送一个“扫描请求”(Scan Request)。如果广播设备被配置为可扫描(Scannable),它将回复一个“扫描响应”包,其中可以包含额外的广播数据,如制造商特定数据(Manufacturer Specific Data)、服务数据(Service Data)等。

在 Android 中,BluetoothLeAdvertiser 提供了 startAdvertisingSet 方法来启动广播集(Advertising Set),它支持更灵活的广播配置,包括传统广播(Legacy Mode)和扩展广播(Extended Advertising)。当使用传统广播模式并期望发送扫描响应时,确保广播设备被设置为“可扫描”至关重要。

2. 核心问题:扫描响应未发送的原因

许多开发者在使用 AdvertisingSet 时,可能会遇到扫描器无法接收到预期扫描响应数据的问题,尤其是在 ScanResult 中发现 mServiceData 或其他预期数据为空。这通常是因为在配置 AdvertisingSetParameters 时,没有明确地将广播设置为可扫描模式。

即使你为 startAdvertisingSet 方法提供了 scanResponse 参数,如果 AdvertisingSetParameters 没有指定 setScannable(true),设备将不会响应扫描请求,从而导致扫描响应数据无法被发送。

3. 解决方案:启用可扫描模式

要确保 AdvertisingSet 能够发送扫描响应,关键在于在构建 AdvertisingSetParameters 时,调用 setScannable(true) 方法。

以下是正确配置 AdvertisingSetParameters 以启用扫描响应的示例代码:

import android.bluetooth.le.AdvertiseData;
import android.bluetooth.le.AdvertisingSetParameters;
import android.bluetooth.le.BluetoothLeAdvertiser;
import android.bluetooth.le.AdvertisingSetCallback;
import android.os.ParcelUuid;
import java.nio.charset.StandardCharsets;

// 假设 serviceId 已经定义,例如:
// private final ParcelUuid serviceId = ParcelUuid.fromString("0000950d-0000-1000-8000-00805f9b34fb");

public class BleAdvertiserManager {

    private BluetoothLeAdvertiser advertiser;
    private AdvertisingSetCallback advertisingSetCallback;

    public BleAdvertiserManager(BluetoothLeAdvertiser advertiser) {
        this.advertiser = advertiser;
        initAdvertisingCallback();
    }

    private void initAdvertisingCallback() {
        advertisingSetCallback = new AdvertisingSetCallback() {
            @Override
            public void onAdvertisingSetStarted(android.bluetooth.le.AdvertisingSet advertisingSet, int txPower, int status) {
                super.onAdvertisingSetStarted(advertisingSet, txPower, status);
                if (status == AdvertisingSetCallback.ADVERTISE_SUCCESS) {
                    // 广播集启动成功
                    // Log.d(TAG, "AdvertisingSet started successfully. TxPower: " + txPower);
                } else {
                    // 广播集启动失败
                    // Log.e(TAG, "AdvertisingSet failed to start. Status: " + status);
                }
            }

            @Override
            public void onAdvertisingDataSet(android.bluetooth.le.AdvertisingSet advertisingSet, int status) {
                super.onAdvertisingDataSet(advertisingSet, status);
                // Log.d(TAG, "Advertising data set. Status: " + status);
            }

            @Override
            public void onScanResponseDataSet(android.bluetooth.le.AdvertisingSet advertisingSet, int status) {
                super.onScanResponseDataSet(advertisingSet, status);
                // Log.d(TAG, "Scan response data set. Status: " + status);
            }

            @Override
            public void onAdvertisingSetStopped(android.bluetooth.le.AdvertisingSet advertisingSet) {
                super.onAdvertisingSetStopped(advertisingSet);
                // Log.d(TAG, "AdvertisingSet stopped.");
            }

            @Override
            public void onAdvertisingEnabled(android.bluetooth.le.AdvertisingSet advertisingSet, boolean enable, int status) {
                super.onAdvertisingEnabled(advertisingSet, enable, status);
                // Log.d(TAG, "Advertising enabled: " + enable + ", status: " + status);
            }
            // 其他回调方法根据需要实现
        };
    }

    public void startBleAdvertising(ParcelUuid serviceId) {
        // 1. 构建 AdvertisingSetParameters
        AdvertisingSetParameters parameters = (new AdvertisingSetParameters.Builder())
                .setConnectable(false) // 设置为不可连接,如果需要可连接则设为true
                .setLegacyMode(true)   // 使用传统广播模式,此模式下扫描响应很重要
                .setScannable(true)    // 关键:允许设备响应扫描请求并发送扫描响应
                .setInterval(AdvertisingSetParameters.INTERVAL_HIGH) // 广播间隔
                .setTxPowerLevel(AdvertisingSetParameters.TX_POWER_MEDIUM) // 发射功率
                .build();

        // 2. 构建主要的 AdvertiseData (广播包数据)
        AdvertiseData data = new AdvertiseData.Builder()
                .setIncludeDeviceName(true) // 包含设备名称
                .setIncludeTxPowerLevel(false) // 不包含发射功率
                .addServiceUuid(serviceId) // 添加服务UUID
                .build();

        // 3. 构建 ScanResponse Data (扫描响应包数据)
        byte[] packetData = "hello_from_scan_response".getBytes(StandardCharsets.UTF_8);
        AdvertiseData response = new AdvertiseData.Builder()
                .setIncludeDeviceName(false) // 扫描响应中不包含设备名称
                .setIncludeTxPowerLevel(false) // 扫描响应中不包含发射功率
                .addServiceUuid(serviceId) // 扫描响应中也添加服务UUID
                .addServiceData(serviceId, packetData) // 关键:在扫描响应中添加服务数据
                .build();

        // 4. 启动 AdvertisingSet
        if (advertiser != null) {
            advertiser.startAdvertisingSet(parameters, data, response, null, null, advertisingSetCallback);
        } else {
            // Log.e(TAG, "BluetoothLeAdvertiser is null. Check Bluetooth adapter state.");
        }
    }

    public void stopBleAdvertising(android.bluetooth.le.AdvertisingSet advertisingSet) {
        if (advertiser != null && advertisingSet != null) {
            advertiser.stopAdvertisingSet(advertisingSetCallback); // 或者直接传入 advertisingSet
        }
    }
}

4. 扫描器验证扫描响应数据

在广播端正确配置并启动广播后,扫描器端需要能够正确解析接收到的 ScanResult。当 setScannable(true) 生效后,ScanResult 中的 ScanRecord 将包含扫描响应中的数据。

以下是扫描器端的示例代码,用于接收并解析扫描结果:

import android.bluetooth.le.BluetoothLeScanner;
import android.bluetooth.le.ScanCallback;
import android.bluetooth.le.ScanFilter;
import android.bluetooth.le.ScanResult;
import android.bluetooth.le.ScanSettings;
import android.os.ParcelUuid;
import android.util.Log;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;

public class BleScannerManager {

    private static final String TAG = "BleScannerManager";
    private BluetoothLeScanner bluetoothLeScanner;
    private ScanCallback scanCallback;

    public BleScannerManager(BluetoothLeScanner scanner) {
        this.bluetoothLeScanner = scanner;
        initScanCallback();
    }

    private void initScanCallback() {
        scanCallback = new ScanCallback() {
            @Override
            public void onScanResult(int callbackType, ScanResult result) {
                super.onScanResult(callbackType, result);
                Log.d(TAG, "ScanResult: " + result.toString());

                // 解析 ScanResult 中的 ScanRecord
                if (result.getScanRecord() != null) {
                    // 获取服务数据
                    Map<ParcelUuid, byte[]> serviceData = result.getScanRecord().getServiceData();
                    if (serviceData != null && !serviceData.isEmpty()) {
                        for (Map.Entry<ParcelUuid, byte[]> entry : serviceData.entrySet()) {
                            ParcelUuid uuid = entry.getKey();
                            byte[] data = entry.getValue();
                            String dataStr = new String(data, StandardCharsets.UTF_8);
                            Log.d(TAG, "Received Service Data - UUID: " + uuid.toString() + ", Data: " + dataStr);
                        }
                    } else {
                        Log.d(TAG, "No Service Data found in ScanRecord.");
                    }

                    // 获取设备名称 (可能在广播包或扫描响应中)
                    String deviceName = result.getScanRecord().getDeviceName();
                    if (deviceName != null) {
                        Log.d(TAG, "Received Device Name: " + deviceName);
                    }
                }
            }

            @Override
            public void onBatchScanResults(List<ScanResult> results) {
                super.onBatchScanResults(results);
                // 批量扫描结果处理,此处省略
            }

            @Override
            public void onScanFailed(int errorCode) {
                super.onScanFailed(errorCode);
                Log.e(TAG, "Scan failed with error code: " + errorCode);
            }
        };
    }

    public void startBleScan(ParcelUuid serviceId) {
        List<ScanFilter> filters = new ArrayList<>();
        ScanFilter filter = new ScanFilter.Builder()
                .setServiceUuid(serviceId) // 根据服务UUID过滤
                .build();
        filters.add(filter);

        // 设置扫描模式为低延迟,以更快地获取扫描结果
        ScanSettings scanSettings = new ScanSettings.Builder()
                .setScanMode(ScanSettings.SCAN_MODE_LOW_LATENCY)
                .build();

        if (bluetoothLeScanner != null) {
            bluetoothLeScanner.startScan(filters, scanSettings, scanCallback);
        } else {
            Log.e(TAG, "BluetoothLeScanner is null. Check Bluetooth adapter state.");
        }
    }

    public void stopBleScan() {
        if (bluetoothLeScanner != null) {
            bluetoothLeScanner.stopScan(scanCallback);
        }
    }
}

通过以上扫描器代码,当广播端正确配置 setScannable(true) 后,扫描器在 onScanResult 回调中,ScanResult 对象的 getScanRecord().getServiceData() 方法将能够成功获取到在扫描响应中添加的 packetData。

ImgGood
ImgGood

免费在线AI照片编辑器

下载

5. 注意事项与最佳实践

  • 权限: 确保你的 Android Manifest 文件中包含了必要的蓝牙权限:

    <uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
    <uses-permission android:name="android.permission.BLUETOOTH" />
    <uses-permission android:name="android.permission.BLUETOOTH_ADVERTISE" />
    <uses-permission android:name="android.permission.BLUETOOTH_SCAN" />
    <uses-permission android:name="android.permission.BLUETOOTH_CONNECT" />
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" /> <!-- Android 6.0+ 需要位置权限进行BLE扫描 -->

    对于 Android 12 (API 31) 及更高版本,需要 BLUETOOTH_ADVERTISE、BLUETOOTH_SCAN 和 BLUETOOTH_CONNECT 权限。同时,对于 BLE 扫描,ACCESS_FINE_LOCATION (或 ACCESS_COARSE_LOCATION) 权限在运行时也需要动态请求。

  • 蓝牙状态: 在启动广播或扫描之前,务必检查设备的蓝牙是否已开启。

  • Legacy Mode 与 Extended Advertising: 示例中使用了 setLegacyMode(true),这意味着广播和扫描响应是两个独立的传统 BLE 广播包。在扩展广播(setLegacyMode(false))中,数据包大小限制有所放宽,但 setScannable(true) 仍然是允许扫描请求/响应交互的关键。如果你的数据量非常大,可以考虑使用扩展广播。

  • 错误处理: 在 AdvertisingSetCallback 和 ScanCallback 中,务必实现 onAdvertisingSetStarted 和 onScanFailed 等错误处理方法,以便及时发现和解决问题。

  • 资源管理: 在不再需要广播或扫描时,务必调用 stopAdvertisingSet() 和 stopScan() 来释放系统资源,避免电量消耗和潜在的连接问题。

  • 数据编码: 确保广播和扫描响应中的自定义数据(如 packetData)在发送和接收时使用相同的编码方式(例如 StandardCharsets.UTF_8),以避免乱码。

总结

在 Android BLE 开发中,使用 AdvertisingSet 进行广播时,如果需要通过扫描响应包传递额外数据,核心在于在 AdvertisingSetParameters.Builder 中显式调用 setScannable(true)。这一简单但关键的配置,能够确保你的设备能够响应扫描请求并成功发送包含丰富信息的扫描响应数据,从而实现更灵活和高效的 BLE 数据广播。同时,配合正确的扫描器端解析逻辑和完善的权限及错误处理,可以构建稳定可靠的 BLE 通信应用。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
android开发三大框架
android开发三大框架

android开发三大框架是XUtil框架、volley框架、ImageLoader框架。本专题为大家提供android开发三大框架相关的各种文章、以及下载和课程。

341

2023.08.14

android是什么系统
android是什么系统

Android是一种功能强大、灵活可定制、应用丰富、多任务处理能力强、兼容性好、网络连接能力强的操作系统。本专题为大家提供android相关的文章、下载、课程内容,供大家免费下载体验。

1821

2023.08.22

android权限限制怎么解开
android权限限制怎么解开

android权限限制可以使用Root权限、第三方权限管理应用程序、ADB命令和Xposed框架解开。详细介绍:1、Root权限,通过获取Root权限,用户可以解锁所有权限,并对系统进行自定义和修改;2、第三方权限管理应用程序,用户可以轻松地控制和管理应用程序的权限;3、ADB命令,用户可以在设备上执行各种操作,包括解锁权限;4、Xposed框架,用户可以在不修改系统文件的情况下修改应用程序的行为和权限。

2140

2023.09.19

android重启应用的方法有哪些
android重启应用的方法有哪些

android重启应用有通过Intent、PendingIntent、系统服务、Runtime等方法。本专题为大家提供Android相关的文章、下载、课程内容,供大家免费下载体验。

284

2023.10.18

Android语音播放功能实现方法
Android语音播放功能实现方法

实现方法有使用MediaPlayer实现、使用SoundPool实现两种。可以根据具体的需求选择适合的方法进行实现。想了解更多语音播放的相关内容,可以阅读本专题下面的文章。

380

2024.03.01

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

88

2026.03.12

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

272

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

59

2026.03.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
SQL 教程
SQL 教程

共61课时 | 4.4万人学习

10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 1.0万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号