我正在构建一个应用程序,它应该使用监听HealthKit
新添加的心率数据的变化,HKObserverQuery
但奇怪的是updateHandler
,当我使用断点和控制台日志进行调试并将新数据手动添加到HealthKit
.
这是大部分代码:
import HealthKit
protocol HeartRateDataProviding {
var delegate: HeartRateDataProviderDelegate? { get set }
func startExecutingQuery(until: Date?)
}
protocol HeartRateDataProviderDelegate: class {
func didQueryData(entry: HeartRateDataEntry)
}
struct HeartRateDataEntry {
let date: Date
let value: Int
}
class HeartRateDataProvider: HeartRateDataProviding {
private let healthStore: HKHealthStore
private let heartRateType = HKQuantityType.quantityType(forIdentifier: HKQuantityTypeIdentifier.heartRate)!
private let heartRateUnit = HKUnit(from: "count/min")
weak var delegate: HeartRateDataProviderDelegate?
init(healthStore: HKHealthStore = .init()) {
self.healthStore = healthStore
}
func startExecutingQuery() {
healthStore.enableBackgroundDelivery(for: heartRateType, frequency: .immediate) { success, error in
print("Observer Query background delivery enabled -> successful: \(success) error: \(String(describing: error))")
}
healthStore.execute(self.createObserverQuery())
}
private func createObserverQuery() -> HKQuery {
let query = HKObserverQuery(sampleType: heartRateType, predicate: nil) { query, completionHandler, error in
print("Hello!")
completionHandler()
}
return query
}
private func formatSamples(samples: [HKSample]?) {
guard let samples = samples as? [HKQuantitySample],
let sample = samples.last else { return }
let entry = HeartRateDataEntry(date: sample.endDate,
value: Int(sample.quantity.doubleValue(for: heartRateUnit)))
delegate?.didQueryData(entry: entry)
}
}
我有一个此类的实例,AppDelegate
并在didFinishLaunchingWithOptions
.
查询的updateHandler
块仅在我设置它并且应用程序进入前台时触发一次,如果我将应用程序置于后台并转到 Health App 并尝试添加新数据,则行为非常不一致。有趣的是,如果我从来没有completionHandler
像苹果文档建议的那样调用它,那么该块会正确触发大约 5-6 次,如果我调用它,completionHandler
那么它可能每 5 次尝试一次就有效。我打算用来HKAnchoredObjectQuery
获取我尝试过的最后一个数据,当观察者触发时它工作正常,但我无法让观察者正常工作。
我错过了什么吗?一些帮助将不胜感激。
更新:出于测试目的,我已将数据类型更改bloodGlucose
为,并注意到updateHandler
每次都能正确触发,这让我认为我实现的代码是正确的,并且行为因数据类型而异,有人可以确认吗?