Ios 如何检索HealthKit记录数组及其元数据

Ios 如何检索HealthKit记录数组及其元数据,ios,swift,healthkit,hksamplequery,Ios,Swift,Healthkit,Hksamplequery,我正在使用swift学习一些关于HealthKit的教程,其中一个教程是如何从HealthKit中检索一些数据,如体重、身高和年龄。本教程介绍了如何检索每个对象的最新记录,以下代码显示: func readMostRecentSample(sampleType:HKSampleType , completion: ((HKSample!, NSError!) -> Void)!) { // 1. Build the Predicate let past = NSDate.distantP

我正在使用swift学习一些关于HealthKit的教程,其中一个教程是如何从HealthKit中检索一些数据,如体重、身高和年龄。本教程介绍了如何检索每个对象的最新记录,以下代码显示:

func readMostRecentSample(sampleType:HKSampleType , completion: ((HKSample!, NSError!) -> Void)!)
{

// 1. Build the Predicate
let past = NSDate.distantPast() as! NSDate
let now   = NSDate()
let mostRecentPredicate = HKQuery.predicateForSamplesWithStartDate(past, endDate:now, options: .None)

// 2. Build the sort descriptor to return the samples in descending order
let sortDescriptor = NSSortDescriptor(key:HKSampleSortIdentifierStartDate, ascending: false)
// 3. we want to limit the number of samples returned by the query to just 1 (the most recent)
let limit = 1

// 4. Build samples query
let sampleQuery = HKSampleQuery(sampleType: sampleType, predicate: mostRecentPredicate, limit: limit, sortDescriptors: [sortDescriptor])
  { (sampleQuery, results, error ) -> Void in

    if let queryError = error {
      completion(nil,error)
      return;
    }

    // Get the first sample
    let mostRecentSample = results.first as? HKQuantitySample

    // Execute the completion closure
    if completion != nil {
      completion(mostRecentSample,nil)
    }
}
// 5. Execute the Query
self.healthKitStore.executeQuery(sampleQuery)
}
然后在另一个类中,开发人员传递参数以获取所需的最新记录,下面的代码显示了检索高度记录的方法:

func updateHeight()
{
// 1. Construct an HKSampleType for Height
let sampleType = HKSampleType.quantityTypeForIdentifier(HKQuantityTypeIdentifierHeight)

// 2. Call the method to read the most recent Height sample
self.healthManager?.readMostRecentSample(sampleType, completion: { (mostRecentHeight, error) -> Void in

  if( error != nil )
  {
    println("Error reading height from HealthKit Store: \(error.localizedDescription)")
    return;
  }

  var heightLocalizedString = self.kUnknownString;
  self.height = mostRecentHeight as? HKQuantitySample;
  // 3. Format the height to display it on the screen
  if let meters = self.height?.quantity.doubleValueForUnit(HKUnit.meterUnit()) {
    let heightFormatter = NSLengthFormatter()
    heightFormatter.forPersonHeightUse = true;
    heightLocalizedString = heightFormatter.stringFromMeters(meters);
  }


  // 4. Update UI. HealthKit use an internal queue. We make sure that we interact with the UI in the main thread
  dispatch_async(dispatch_get_main_queue(), { () -> Void in
    self.heightLabel.text = heightLocalizedString
    self.updateBMI()
  });
})

}
我创建了一个与第一个方法类似的方法,但改动很少,因此我可以获得一个包含10个葡萄糖记录的数组:

func readAllGlucose(sampleType:HKSampleType , completion: (([HKSample!], NSError!) -> Void)!)
{

    let now   = NSDate()
    let df = NSDateFormatter()
    df.dateFormat = "yyyy-MM-dd"
    let pastt = df.dateFromString("2015-05-18")

    //let mostRecentPredicate = HKQuery.predicateForSamplesWithStartDate(past, endDate:now, options: .None)
    let allreadings = HKQuery.predicateForSamplesWithStartDate(pastt, endDate: now, options: .None)

    // 2. Build the sort descriptor to return the samples in descending order
    let sortDescriptor = NSSortDescriptor(key:HKSampleSortIdentifierStartDate, ascending: false)
    // 3. we want to limit the number of samples returned by the query to just 1 (the most recent)
    let limit = 10

    // 4. Build samples query

    let sampleQuery = HKSampleQuery(sampleType: sampleType, predicate: allreadings, limit: limit, sortDescriptors: [sortDescriptor])
        { (sampleQuery, results, error ) -> Void in

            if let queryError = error {
                completion([nil],error)
                return;
            }

            // Get the first sample
            let allSamples = results as? [HKQuantitySample]

            // Execute the completion closure
            if completion != nil {
                completion(allSamples!,nil)
            }
    }
    // 5. Execute the Query
    self.healthKitStore.executeQuery(sampleQuery)
}
然后,我创建了另一个类似于updateHeight()方法的方法,但当然要做必要的更改:

func updateLastGlucoRecords()
{
    // 1. Construct an HKSampleType for weight
    let sampleType = HKSampleType.quantityTypeForIdentifier(HKQuantityTypeIdentifierBloodGlucose)

    // 2. Call the method to read the most recent weight sample
    self.healthManager?.readAllGlucose(sampleType, completion: {([allReadings], error) -> Void in

        if (error != nil) {
            println("Error reading glucose readings from HealthKit Store: \(error.localizedDescription)")
            return;
        }

        var glucoseLocalizedString = self.kUnknownString;
        self.glucose = allReadings as? [HKQuantitySample]
        for reading in readings {

            if let record = reading.quantity {
                glucoseLocalizedString = record
            }
            dispatch_async(dispatch_get_main_queue(), { () -> Void in
                self.glucoReadings.append("\(glucoseLocalizedString)")
            })

        }


    })
    self.healthManager?.readMostRecentSample(sampleType, completion: { (mostRecentWeight, error) -> Void in

        if( error != nil )
        {
            println("Error reading weight from HealthKit Store: \(error.localizedDescription)")
            return;
        }

        var weightLocalizedString = self.kUnknownString;
        // 3. Format the weight to display it on the screen
        self.weight = mostRecentWeight as? HKQuantitySample;
        if let kilograms = self.weight?.quantity.doubleValueForUnit(HKUnit.gramUnitWithMetricPrefix(.Kilo)) {
            let weightFormatter = NSMassFormatter()
            weightFormatter.forPersonMassUse = true;
            weightLocalizedString = weightFormatter.stringFromKilograms(kilograms)
        }

        // 4. Update UI in the main thread
        dispatch_async(dispatch_get_main_queue(), { () -> Void in
            self.weightLabel.text = weightLocalizedString
            self.updateBMI()

        });
    });
}
但不幸的是,我在线路上遇到了两个错误:

self.healthManager?.readAllGlucose(sampleType, completion: {([allReadings], error) -> Void in
第一个错误:

[HKSample!]'不是'

第二个错误:

使用未声明的类型“AllReaders”

我还有一个问题,如何从每个对象获取元数据,我想获取插入数据的日期和时间

如果有人有任何可以解决这个问题的想法,我将非常感激


提前谢谢

我也有同样的问题。请查看下面我的代码。 1.所有读数错误的原因是您尚未通过HKSample数组,因此您必须写入:让所有读数:[HKSample!]然后通过所有读数

func readAllSample(sampleType:HKSampleType , completion: (([HKSample!], NSError!) -> Void)!)
{
    let now   = NSDate()
    let past = now.dateByAddingTimeInterval(-10*24*60*60) as NSDate!
    let mostRecentPredicate = HKQuery.predicateForSamplesWithStartDate(past, endDate:now, options: .None)
    let sortDescriptor = NSSortDescriptor(key:HKSampleSortIdentifierStartDate, ascending: false)
    let limit = 10
    let sampleQuery = HKSampleQuery(sampleType: sampleType, predicate: mostRecentPredicate, limit: limit, sortDescriptors: [sortDescriptor])
        { (sampleQuery, results, error ) -> Void in

            if(error == nil)
            {
                print(results)
            }

    }
    self.healthKitStore.executeQuery(sampleQuery)
}
可能重复的