20

精简版:

在我对 Android 5.0 Lollipop 的测试中,我注意到android.bluetooth.le.BluetoothLeScanner检测 BLE 设备的频率低于 Android 4.4 KitKat。为什么会这样,还有其他选择吗?

长版:

我正在开发一个 Android 应用程序,专门用于 Nexus 7 平板电脑,专注于检测蓝牙低功耗 (BLE) 设备。该应用程序主要对信标的 RSSI 值感兴趣,以确定它们与平板电脑的接近程度。这意味着我不需要连接到 BLE 设备,因为当检测到设备时,RSSI 值会传递给扫描回调。

在 Android 4.4 KitKat 中,当我调用时BluetoothAdapter.startLeScan(LeScanCallback),我的回调只会为每个检测到的 BLE 设备调用一次。(我已经看到一些讨论声称这种行为可能因设备而异)但是,我对不断变化的 RSSI 值感兴趣,因此目前推荐的方法是以设定的间隔(在我的情况下为 250 毫秒)连续执行 startLeScan 和 stopLeScan:

public class TheOldWay {

    private static final int SCAN_INTERVAL_MS = 250;

    private Handler scanHandler = new Handler();
    private boolean isScanning = false;

    public void beginScanning() {
        scanHandler.post(scanRunnable);
    }

    private Runnable scanRunnable = new Runnable() {
        @Override
        public void run() {
            BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();

            if (isScanning) {
                adapter.stopLeScan(leScanCallback);
            } else if (!adapter.startLeScan(leScanCallback)) {
                // an error occurred during startLeScan
            }

            isScanning = !isScanning;

            scanHandler.postDelayed(this, SCAN_INTERVAL_MS);
        }
    };

    private BluetoothAdapter.LeScanCallback leScanCallback = new BluetoothAdapter.LeScanCallback() {
        @Override
        public void onLeScan(BluetoothDevice device, int rssi, byte[] scanRecord) {
            // use the RSSI value
        }
    };

}

本质上,这给了我所需的结果,但这个过程非常耗费资源,并最终导致蓝牙适配器无响应。

由于这些原因,我将我的 Nexus 7 升级到了 Android 5.0 Lollipop,看看我的 BLE 问题是否会得到修复。在 Lollipop 中,BluetoothAdapter.startLeScan(LeScanCallback) 已被弃用,取而代之的是允许对扫描过程进行更多控制的新 API 。从我的第一次测试来看,当 RSSI 值发生变化时,startScan 似乎不会连续调用我的回调(在我的 Nexus 7 上),所以我仍然需要使用 startScan / stopScan 实现:

@TargetApi(21)
public class TheNewWay {

    private static final int SCAN_INTERVAL_MS = 250;

    private Handler scanHandler = new Handler();
    private List<ScanFilter> scanFilters = new ArrayList<ScanFilter>();
    private ScanSettings scanSettings;
    private boolean isScanning = false;

    public void beginScanning() {
        ScanSettings.Builder scanSettingsBuilder = new ScanSettings.Builder();
        scanSettingsBuilder.setScanMode(ScanSettings.SCAN_MODE_LOW_LATENCY);
        scanSettings = scanSettingsBuilder.build();

        scanHandler.post(scanRunnable);
    }

    private Runnable scanRunnable = new Runnable() {
        @Override
        public void run() {
            BluetoothLeScanner scanner = BluetoothAdapter.getDefaultAdapter().getBluetoothLeScanner();

            if (isScanning) {
                scanner.stopScan(scanCallback);
            } else {
                scanner.startScan(scanFilters, scanSettings, scanCallback);
            }

            isScanning = !isScanning;

            scanHandler.postDelayed(this, SCAN_INTERVAL_MS);
        }
    };

    private ScanCallback scanCallback = new ScanCallback() {
        @Override
        public void onScanResult(int callbackType, ScanResult result) {
            super.onScanResult(callbackType, result);

            int rssi = result.getRssi();

            // do something with RSSI value
        }

        @Override
        public void onScanFailed(int errorCode) {
            super.onScanFailed(errorCode);

            // a scan error occurred
        }
    };

}

如您所见,我已经使用 ScanSettings 类配置了扫描仪,它允许您设置scanMode. 我使用ScanSettings.SCAN_MODE_LOW_LATENCY,它具有以下文档:“使用最高占空比进行扫描。建议仅在应用程序在前台运行时使用此模式。” 听起来和我想要的完全一样,但不幸的是,我每 15 - 30 秒检测一次信标,而 KitKat 版本在此扫描间隔内每 1 - 2 秒显示一次相同的信标。

您知道造成这种差异的原因是什么吗?我错过了什么,也许是一些新的设置?是否有其他方法可以完成上述操作?

提前非常感谢!

亚伯

PS:我想包含更多指向我使用过的资源的链接,但我还没有代表点。

4

5 回答 5

6

使用运行新的 Android 5.0 扫描 API 的 Nexus 5,我得到了非常不同的结果。使用 SCAN_MODE_LOW_LATENCY 时,BLE 数据包的检测几乎是实时的,对于以 10Hz 传输的 BLE 信标,每 100 毫秒。

你可以在这里阅读完整的结果:

http://developer.radiusnetworks.com/2014/10/28/android-5.0-scanning.html

这些测试基于在此处运行开源 Android Beacon Library 2.0 的实验性 android-l-apis 分支。

测试结果的差异并不明显,但开始和停止扫描可能会改变结果。

编辑:硬件可能是不同的。在 Nexus 4 上查看类似时间的报告:https ://github.com/AltBeacon/android-beacon-library/issues/59#issuecomment-64281446

于 2014-11-24T20:41:41.250 回答
5

我的评论还没有 50 声望,所以请耐心等待,这条评论将以答案的形式出现。在您的代码中,这部分不应该:

if (isScanning) { scanner.startScan(...)

改为:

if (!isScanning) {
 scanner.startScan(...)

因为按照您的代码,您在开始扫描之前调用了 stopScan() 。如果 stopScan() 方法是幂等/安全的,它可能不会对结果产生直接影响。但是您知道,为了代码的可理解性,您应该编辑问题。并对你的代码做同样的事情,有时拜占庭的事情在起作用;)

您是否为 SCAN_INTERVAL_MS 尝试过更大的值?如果是,有多大?

于 2015-02-26T16:00:50.130 回答
2

我的 Nexus 4 在 KitKat 和 Lollipop 中都经历了非常相似的结果。

使用 KitKat 时,蓝牙适配器最终也没有响应;起初我认为它可能与较短的扫描间隔(200ms)有关,但将该数字增加到甚至一秒钟并没有帮助,在这件事上,我发现,当以编程方式禁用和启用适配器时,有时可以解决问题. 不幸的是,我不能说它一直有效。

现在有了 Lollipop,我寄予厚望来解决这个问题,我遇到了与您描述的相同的行为。我还必须使用 startScan / stopScan 实现,在检测时间方面得到类似的结果。可悲的是,我还没有找到解决方法来更快地获得结果。

根据您的描述,我认为这可能是硬件问题,即使 Nexus 7 和 Nexus 4 来自不同的制造商(华硕和 LG)。

我知道除了试图回答您关于您遗漏某些东西的问题之外,我在这里并没有提供太多帮助;我不这么认为,我认为问题出在硬件或蓝牙 API 之类的东西上,它们在不同设备上的表现仍然不尽相同。

于 2014-11-25T15:50:21.407 回答
1

超过 API 21 android 默认使用SCAN_MODE_LOW_POWERSCAN_MODE_LOW_POWER

尝试 SCAN_MODE_BALANCED 看看它是否会变得更好。
SCAN_MODE_BALANCED

于 2016-01-27T12:52:05.860 回答
0

如果您BW13_DayOne_Session1 Bluetooth Advanced在 google 上搜索,您会找到一个 pdf 文档,该文档会根据发现设置为您提供设备延迟(参见第 8 页)。我猜你的问题与这些时间有关。您可以通过确定您正在测试的设备的广告配置(Adv Int、占空比)来进行验证,然后确定 API 设置在配置扫描间隔等方面所做的工作。一旦有了这些,您就可以使用该表进行插值以查看您是否获得了预期的结果。

我知道这是一个软件站点,但通常在与硬件接口时,您需要了解协议,否则您会在黑暗中拍摄。

于 2015-02-27T19:07:58.337 回答