在我的应用程序中,我有开始和停止按钮,当用户按下开始时,我调用startScan方法
bluetoothAdapter.getBluetoothLeScanner().startScan(getLeScanCallback());
当用户按下停止时,我调用stopScan,但它似乎没有做任何事情。蓝牙适配器不断扫描新设备。
bluetoothAdapter.getBluetoothLeScanner().stopScan(getLeScanCallback());
这是我的getLeScanCallback方法:
private ScanCallback getLeScanCallback(){
ScanCallback leScanCallback = new ScanCallback() {
@Override
public void onScanResult(int callbackType, ScanResult result) {
super.onScanResult(callbackType, result);
boolean duplicate = false;
for(Device d : devices) {
if(d.getAddress().equals(result.getDevice().getAddress()))
duplicate = true;
}
if(!duplicate) {
Device device = new Device();
device.setName(result.getDevice().getName());
device.setAddress(result.getDevice().getAddress());
device.setStatus(getString(R.string.disconnected));
devices.add(device);
deviceListAdapter.notifyDataSetChanged();
}
}
@Override
public void onBatchScanResults(List<ScanResult> results) {
super.onBatchScanResults(results);
}
@Override
public void onScanFailed(int errorCode) {
super.onScanFailed(errorCode);
}
};
return leScanCallback;
}
即使在调用 stopScan() 之后也会调用它。我做错了什么,换句话说,如何停止扫描BLE设备?
每次调用getLeScanCallback
时,都会创建ScanCallback
的新实例并丢失对先前实例的引用。
无论您通过哪个实例,stopScan
都会停止正在进行的扫描,但如果它是不同的实例(而不是您用于启动的实例),它不会删除回调的旧实例,因此在停止之前仍然会向它传递一些事件。
扫描结果不会立即停止,但最终会停止。有一个待处理结果队列,您可以使用这些结果进行刷新:
scanner.flushPendingScanResults();
但即便如此,文档也很清楚它会将找到的任何内容发送到回调中。即使您已停止扫描,也会发生这种情况。糟糕的 API 设计,但它就是这样。
我认为最简单的方法是设置一个标志以在停止扫描结果后忽略扫描结果。
我没有注意到你的代码有任何问题,似乎很好。也许您的getLeScanCallback()
方法有问题?
您可以将其声明为成员变量,以便leScanCallback = new ScanCallback()
,然后您可以在尝试停止扫描时引用此成员变量,而不是使用返回 ScanCallback 的方法。
bluetoothAdapter.getBluetoothLeScanner().stopScan(leScanCallback);
根据您支持的Android版本,添加Rajesh Panchal为Lollipop之前的Android设备建议的startLeScan
和stopLeScan
方法可能是值得的。
替换
bluetoothAdapter.getBluetoothLeScanner().startScan(getLeScanCallback());
bluetoothAdapter.getBluetoothLeScanner().stopScan(getLeScanCallback());
跟
bluetoothAdapter.getBluetoothLeScanner().startLeScan(getLeScanCallback());
bluetoothAdapter.getBluetoothLeScanner().stopLeScan(getLeScanCallback());