gpt4 book ai didi

ios - 如何将 HKStatisticsQuery 的结果返回到 Swift 2 中的变量?

转载 作者:行者123 更新时间:2023-11-29 01:16:45 27 4
gpt4 key购买 nike

已于美国东部时间 2016 年 1 月 30 日晚上 7:40 更新于底部

所以我尝试运行一个 StatisticsQuery 来获取 HealthKit 中存储的当天的总 DistanceRunningWalking,然后将该查询的结果存储在一个变量中,以供稍后使用。查询似乎工作正常,因为我已经测试过将查询结果 (totalDistance) 从函数内打印到标签。我遇到的问题是尝试将结果保存到变量时。

这是我的 HealthKitManager.swift 文件中的代码:

import HealthKit

class HealthKitManager {

class var sharedInstance: HealthKitManager {
struct Singleton {
static let instance = HealthKitManager()
}

return Singleton.instance
}

let healthStore: HKHealthStore? = {
if HKHealthStore.isHealthDataAvailable() {
return HKHealthStore()
} else {
return nil
}
}()

let distanceCount = HKQuantityType.quantityTypeForIdentifier(HKQuantityTypeIdentifierDistanceWalkingRunning)

let distanceUnit = HKUnit(fromString: "mi")

}

ViewController 顶部的代码:(这是我要保存到的变量)

let healthKitManager = HealthKitManager.sharedInstance

//Set up variable to contain result of query
var distanceTotalLength:Double?

viewDidLoad中的代码:

//Run the function
requestHealthKitAuthorization()

//Set value of variable to query result
distanceTotalLength = queryDistanceSum()

ViewController 主体中的代码:

func requestHealthKitAuthorization() {
let dataTypesToRead = NSSet(objects: healthKitManager.distanceCount!)
healthKitManager.healthStore?.requestAuthorizationToShareTypes(nil, readTypes: dataTypesToRead as NSSet as? Set<HKObjectType>, completion: { [unowned self] (success, error) in
if success {
self.queryDistanceSum()
} else {
print(error!.description)
}
})
}

func queryDistanceSum() {
let sumOption = HKStatisticsOptions.CumulativeSum
let startDate = NSDate().dateByRemovingTime()
let endDate = NSDate()
let predicate = HKQuery.predicateForSamplesWithStartDate(startDate, endDate: endDate, options: [])

let statisticsSumQuery = HKStatisticsQuery(quantityType: healthKitManager.distanceCount!, quantitySamplePredicate: predicate, options: sumOption) {
[unowned self] (query, result, error) in
if let sumQuantity = result?.sumQuantity() {
dispatch_async(dispatch_get_main_queue(), {
let totalDistance = sumQuantity.doubleValueForUnit(self.healthKitManager.distanceUnit)
self.distanceTotalLength = totalDistance
})
}
}
healthKitManager.healthStore?.executeQuery(statisticsSumQuery)
}

在最后一行 return (distanceTotalLength)! 启动应用程序时出现错误,显示为 fatal error: expectedly found nil while unwrapping an Optional value。我已经意识到这很可能是处理范围的问题(尽管我可能做错了其他事情,所以请指出任何问题)但我自己没有看到/找到问题的解决方案。

任何有关此问题的帮助将不胜感激,所以提前致谢!

更新:2016 年 1 月 30 日 @ 美国东部时间下午 7:40 左右

好的,所以我一直在尝试自己修复这个问题,并且我发现了:代码肯定可以工作。通过将 0 的初始值分配给 ViewController 顶部的 distanceTotalLength,我能够运行应用程序而不会出现 fatal error 。然而,当我随后尝试通过 prepareForSegue 函数将 distanceTotalLength 的值传递给另一个 View 时,我意识到它毕竟是被分配的。当我转到该 View 时,它没有使用 0 的初始值,而是使用查询的结果。

我测试它的方法是设置变量:var distanceTotalLength:Double = 0 在我的 viewController 的最顶部,在 viewDidLoad 之前 然后在 viewDidLoad 我使用 distanceLabel.text = String(distanceTotalLength) 将值分配给标签,正如我所说,标签最终显示为 0。但是,当我转换到另一个 View 、传递 distanceTotalLength 的值并打印该值时,它就可以工作了。在第二个屏幕上,它打印查询结果,而不是 0

所以我假设问题是查询运行,然后在 View 已经加载所有预定义值之后分配值。不幸的是,这是我再次陷入困境的地方。既然我已经走到这一步了,有人知道如何帮助我吗?

最佳答案

你是对的。在您的 HKStatisticsQuery 完成执行后调用的 completionHandler 闭包发生在稍后被破坏的时间。将查询执行想象成在邮件中向某人发送一封信,然后等待他们的回复;这不会是立竿见影的,但与此同时您可以离开并做其他事情。

要在您的代码中处理此问题,请将您自己的完成闭包添加到 queryDistanceSum 方法。然后在设置 self.distanceTotalLength = totalDistance 之后,调用该闭包。在实现闭包代码时,添加在设置距离后需要完成的任何操作,例如更新您的 UI。

func queryDistanceSum(completion: () -> Void) { // <----- add the closure here
let sumOption = HKStatisticsOptions.CumulativeSum
let startDate = NSDate().dateByRemovingTime()
let endDate = NSDate()
let predicate = HKQuery.predicateForSamplesWithStartDate(startDate, endDate: endDate, options: [])

let statisticsSumQuery = HKStatisticsQuery(quantityType: healthKitManager.distanceCount!, quantitySamplePredicate: predicate, options: sumOption) {
[unowned self] (query, result, error) in
if let sumQuantity = result?.sumQuantity() {
dispatch_async(dispatch_get_main_queue(), {
let totalDistance = sumQuantity.doubleValueForUnit(self.healthKitManager.distanceUnit)
self.distanceTotalLength = totalDistance
completion() // <----- call the closure here
})
}
}
healthKitManager.healthStore?.executeQuery(statisticsSumQuery)
}


// Then whenever you need to update the distance sum call the function
// with the closure, then handle the result as needed
queryDistanceSum { () -> () in
// distanceTotalLength has now been set.
// Update UI for new distance value or whatever you need to do
}

无论何时实现闭包,您都必须假设闭包中的代码将在稍后执行。

关于ios - 如何将 HKStatisticsQuery 的结果返回到 Swift 2 中的变量?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35092153/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com