首页 > 解决方案 > CLLocationManager 的委托函数“didRangeBeacon”没有找到任何信标?

问题描述

这个问题已经在这里问过了,但它并没有解决我的问题,我会在问题的最后解释为什么它不能解决问题。

我正在做的需要两个步骤:

  1. 让 CBCentralManager 发现外设

  2. 让 CLLocationManager 监听在步骤 1 中发现的信标

我使用以下委托方法找到了我的信标的 UUID:

func centralManager(_ central: CBCentralManager, didDiscover peripheral: CBPeripheral, advertisementData: [String: Any], rssi RSSI: NSNumber) {
    print(peripheral.identifier)
}

然后进行第 2 步:

if let uuid = UUID.init(uuidString: "<identifier found in step 1>") {
    locationManager.startRangingBeacons(in: .init(proximityUUID: uuid, identifier: UUID().uuidString))
}

我的 locationManager 的委托方法只是打印范围内的所有信标:

func locationManager(_ manager: CLLocationManager, didRangeBeacons beacons: [CLBeacon], in region: CLBeaconRegion) {
    print(beacons)
}

但我得到的只是一个空数组。为什么?我该如何解决?

上面列出的问题表明,在创建 时CLBeaconRegion,我们应该给出一个唯一标识符,我已经在这样做了。

标签: iosswiftbeaconcbperipheral

解决方案


问题出在这句话上:

我使用以下委托方法找到了我的信标的 UUID:

 func centralManager(_ central: CBCentralManager, didDiscover peripheral: CBPeripheral, advertisementData: [String: Any], rssi RSSI: NSNumber) {
     print(peripheral.identifier)
 }

返回的值peripheral.identifier不是您的信标的proximityUUID. 因此,didRangeBeacons返回一个空数组,因为附近没有与您设置的proxmityUUID匹配的信标。CLBeaconRegion

这里有一个更深层次的解释:

peripheral.identifier字段来自CBPeripheral对象,该对象继承自CBPeer. Apple 的文档是这样描述的:

此属性的值表示对等​​方的唯一标识符。本地管理器第一次遇到对等点时,系统会为对等点分配一个 UUID,由新的 NSUUID 对象表示。对等点由 NSUUID UUID 标识,而不是由标识外围设备的服务、特征和特征描述符的 CBUUID 对象标识。

基本上,此字段只是 iOS 分配给蓝牙设备的临时标识符。虽然它由 a 表示NSUUID(在 Objective C 中或仅UUID在 Swift 中调用),并且CLBeacon/CLBeaconRegion有一个proximityUUID也是 的实例的字段NSUUID但它们的值不同。 这是一个常见的混淆来源。

不幸的是,不可能使用 CoreBluetooth API 来获取信标的 ProximityUUID。Apple 故意不遗余力地阻止您这样做。对不起。


推荐阅读