【编者按】本文作者为 Matthew Maher,文章手把手地介绍了如何借助 HealthKit 建立简单的健身应用,包含诸多代码实例。本文系国内 ITOM 管理平台 OneAPM 编译呈现。

根据新闻报导,健康与健美在今时今日的重要程度比已往任何时候都高。说起来有点可笑,似乎就在几天之前,笔者就见到过类似的新闻。或许,这是当人逐渐变老之后挥之不去的感觉吧——渴望保持健康以及健美的感觉。不管怎么说,健康与健美是一个重要话题。技术的进步,尤其是移动应用与硬件世界的不断提高,正为这个似乎日益成长的话题带来全新的契机。

HealthKit 是苹果公司推出的一款移动应用平台,旨在为重要、可追踪的健康数据与注重健康、热衷锻炼的科技消费者搭起桥梁。这很酷。用户可以轻松地追踪一段时间内可测量的健身与健康数据。除了了解自身的健康数据,看到图表中喜人的增长曲线也的确鼓舞人心。

正如人们想象的那样,在管理健康信息时安全是非常重要的考虑因素。HealthKit 直截了当地将所有 HealthKit 信息的绝对控制权置于用户的手中。用户可以授权或拒绝任何应用对其健康数据发出的读取请求。

作为开发者,我们需要征求许可才能从/向 HealthKit 读取/写入数据。实际上,我们需要明确地声明打算读取或改变的数据。此外,任何使用 HealthKit 的应用都必须包含隐私政策,这样一来,用户才能对其信息的处理感到更加放心。

关于 OneHourWalker

在本文中,我们将打造一个有趣的小应用,它会从 HealthKit 读取数据,也会向其写入新数据。来见一见 OneHourWalker 吧。

OneHourWalker 是一款追踪使用者在一个小时内行走或跑步之距离的健身应用。用户可以将距离与 HealthKit 分享,之后就能在健康应用中读取之。我知道,一个小时听起来有点过于乐观了(至少笔者本人可能无法坚持下去)。因此,用户也可以提早中止计数,并分享距离。

额,到目前为止,似乎 OneHourWalker 只会向 HealthKit 写入数据。我们需要读取什么数据呢?

好问题!在步行锻炼时,我喜欢选择乡间或林间小路。常常,我会遇到树枝低垂的区域。而我是一条身高 193cm 的汉子,这真的让我很苦恼。解决办法是:从 HealthKit 读取用户的身高数据,将之打印为应用的一个标签。这个标签可以作为对用户的善意提醒,这样,他们就能避免在步行时被树枝打到。

首先,点此下载 OneHourWalker 的初始项目。先试着跑起来,找找应用运行的感觉。计数器与地点追踪功能已经在运行了,所以我们只需专注于 HealthKit 实现。注意,当到达 60 分钟时间点时,计算器与追踪都会停止。

启用 HealthKit

首先,在我们的应用中启用 HealthKit。在项目导航中,点击 OneHourWalker,之后点击 Targets 下面的 OneHourWalker,之后选择屏幕顶部的 Capabilities 选项。

查看 Capabilities 列表的底部,启用 HealthKit。这一简单的操作会将 HealthKit 权限添加到 App ID,将 HealthKit 键添加到 info plist 文件,将 HealthKit 权限添加到授权文件,并且与 HealthKit.framework 相连接。就是这么简单。

开始编程

接下来,跳转到 TimerViewController.swift,开始将 HealthKit 引入 OneHourWalker。首先,创建一个 HealthKitManager 实例。

import UIKit
import CoreLocation
import HealthKitclass TimerViewController: UIViewController, CLLocationManagerDelegate {@IBOutlet weak var timerLabel: UILabel!@IBOutlet weak var milesLabel: UILabel!@IBOutlet weak var heightLabel: UILabel!    var zeroTime = NSTimeInterval()    var timer : NSTimer = NSTimer()let locationManager = CLLocationManager()    var startLocation: CLLocation!    var lastLocation: CLLocation!    var distanceTraveled = 0.0let healthManager:HealthKitManager = HealthKitManager()

所有 HealthKit 工作都会在 HealthKitManager.swift 中进行。它会包含重要的方法,我们很快就会谈到。

正如在前文介绍部分所述,我们需要取得用户的许可,才能读取并修改他们的健康数据。在 viewDidLoad()中,我们就得这么做。

    override func viewDidLoad() {super.viewDidLoad()locationManager.requestWhenInUseAuthorization()if CLLocationManager.locationServicesEnabled(){locationManager.delegate = selflocationManager.desiredAccuracy = kCLLocationAccuracyBest} else {print("Need to Enable Location")}// We cannot access the user's HealthKit data without specific permission.getHealthKitPermission()
}

getHealthKitPermission() 方法会调用 manager 的 authorizeHealthKit()方法。如果一切顺利,我们便能调用setHeight()方法。不过,我们很快会在后文中谈到此方法。

func getHealthKitPermission() {        // Seek authorization in HealthKitManager.swift.healthManager.authorizeHealthKit { (authorized,  error) -> Void in                        if authorized {                // Get and set the user's height.self.setHeight()} else {                if error != nil {                    print(error)}                print("Permission denied.")}}
}

在 HealthKitManager.swift 中,我们会创建 authorizeHealthKit() 方法。然而,除此之外,我们需要创建 HealthKit 存储,用于连接应用与 HealthKit 的数据。

let healthKitStore: HKHealthStore = HKHealthStore()func authorizeHealthKit(completion: ((success: Bool, error: NSError!) -> Void)!) {// State the health data type(s) we want to read from HealthKit.        let healthDataToRead = Set(arrayLiteral: HKObjectType.quantityTypeForIdentifier(HKQuantityTypeIdentifierHeight)!)// State the health data type(s) we want to write from HealthKit.        let healthDataToWrite = Set(arrayLiteral: HKObjectType.quantityTypeForIdentifier(HKQuantityTypeIdentifierDistanceWalkingRunning)!)// Just in case OneHourWalker makes its way to an iPad...        if !HKHealthStore.isHealthDataAvailable() {            print("Can't access HealthKit.")}// Request authorization to read and/or write the specific data.        healthKitStore.requestAuthorizationToShareTypes(healthDataToWrite, readTypes: healthDataToRead) { (success, error) -> Void in            if( completion != nil ) {completion(success:success, error:error)}}
}

在请求获取用户健康数据的授权时,我们需要明确指定打算读取以及修改的信息。对本例而言,我们需要读取用户的身高,从而帮助他们躲避有危险的低垂枝丫。我们希望 HealthKit 能提供一个可以转化为可理解的身高的 HKObject 量。此外,我们还要获得修改 HKObject 量的许可,以记录用户的行走及跑步距离。

在处理好 OneHourWalker 与 iPad 通信的可能性后,我们做出官方请求。

HealthKitManager.swift 中,创建从 HealthKit 读取用户身高数据的 getHeight() 方法。

func getHeight(sampleType: HKSampleType , completion: ((HKSample!, NSError!) -> Void)!) {// Predicate for the height query        let distantPastHeight = NSDate.distantPast() as NSDate        let currentDate = NSDate()        let lastHeightPredicate = HKQuery.predicateForSamplesWithStartDate(distantPastHeight, endDate: currentDate, options: .None)// Get the single most recent height        let sortDescriptor = NSSortDescriptor(key: HKSampleSortIdentifierStartDate, ascending: false)// Query HealthKit for the last Height entry.        let heightQuery = HKSampleQuery(sampleType: sampleType, predicate: lastHeightPredicate, limit: 1, sortDescriptors: [sortDescriptor]) { (sampleQuery, results, error ) -> Void in                if let queryError = error {completion(nil, queryError)                    return}                // Set the first HKQuantitySample in results as the most recent height.                let lastHeight = results!.first                if completion != nil {completion(lastHeight, nil)}}        // Time to execute the query.self.healthKitStore.executeQuery(heightQuery)
}

查询身高数据的第一步是创建一个断言以定义时间参数。我们是在请求一段时间内的所有身高数据——与当前日期相距甚远的一个过去的日期。显然,这会返回一个数组。然而,我们只想要最近期的身高,因此,我们请求数据时可以让最新的数据排在数组的最前头。

在构建这一查询时,我们会把数组的长度限制为1。在考虑好出现错误的可能性后,我们会将结果中的首个也即唯一一个数组项目分配给 lastHeight。接下来,完善 getHeight() 方法。最后,针对用户的健康数据执行查询。

回到 TimerViewController.swift,在 app 真正投入使用之前,假设用户授权了适当的许可,则 setHeight() 方法会被 getHealthKitPermission() 调用。

var height: HKQuantitySample?

首先,我们需要为 HKQuantitySample 实例声明一个身高变量。

func setHeight() {        // Create the HKSample for Height.        let heightSample = HKSampleType.quantityTypeForIdentifier(HKQuantityTypeIdentifierHeight)        // Call HealthKitManager's getSample() method to get the user's height.              self.healthManager.getHeight(heightSample!, completion: { (userHeight, error) -> Void in            if( error != nil ) {print("Error: \(error.localizedDescription)")                return}            var heightString = ""            self.height = userHeight as? HKQuantitySample            // The height is formatted to the user's locale.            if let meters = self.height?.quantity.doubleValueForUnit(HKUnit.meterUnit()) {                let formatHeight = NSLengthFormatter()formatHeight.forPersonHeightUse = trueheightString = formatHeight.stringFromMeters(meters)}            // Set the label to reflect the user's height.dispatch_async(dispatch_get_main_queue(), { () -> Void in                                self.heightLabel.text = heightString})})}

share() 方法之上,我们会创建 setHeight() 方法。我们请求的身高数据样本以 HKQuantity 返回,标识符 HKQuantityTypeIdentifierHeight 知道这一对象。

接下来,调用在 manager 中创建的 getHeight() 方法。有了身高样本,我们还需要将之翻译为恰当的字符串以展示在标签中。与往常一样,考虑所有可能的错误情况是很重要的。

到此,用户就可以打开 app,查看他们的身高(如果他的健康应用中记录着身高数据),开启计时器,追踪他跑步或行走的距离了。接下来,我们要处理将距离数据写入健康应用的过程,这样,用户才能在同一个应用中保存其所有的健身数据。

在用户结束外出锻炼之后,不管有没有到60分钟,他可能会使用 Share(分享)按钮将其辛苦赚得的运动距离发送到健康应用。所以,在 share() 方法中,我们需要调用 HealthKitManager.swiftsaveDistance() 方法来实现这一过程。在这个方法中,我们会发送运动距离以及取得该距离的日期。这样,用户便能在第二天争取更好的成绩。

@IBAction func share(sender: AnyObject) {    healthManager.saveDistance(distanceTraveled, date: NSDate())
}

接下来,回到 manager,我们要在此处创建 saveDistance() 方法。首先,我们要让 HealthKit 知道我们打算写入一个代表步行及跑步距离的量。之后,将度量单位设置为英里,并赋值官方的样本量。HealthKit 的 saveObject() 方法会将此数据写入用户的健康数据。

func saveDistance(distanceRecorded: Double, date: NSDate ) {        // Set the quantity type to the running/walking distance.let distanceType = HKQuantityType.quantityTypeForIdentifier(HKQuantityTypeIdentifierDistanceWalkingRunning)        // Set the unit of measurement to miles.let distanceQuantity = HKQuantity(unit: HKUnit.mileUnit(), doubleValue: distanceRecorded)        // Set the official Quantity Sample.let distance = HKQuantitySample(type: distanceType!, quantity: distanceQuantity, startDate: date, endDate: date)        // Save the distance quantity sample to the HealthKit Store.healthKitStore.saveObject(distance, withCompletion: { (success, error) -> Void in            if( error != nil ) {                print(error)} else {                print("The distance has been recorded! Better go check!")}})
}

跳转到健康应用,所记录的数据会出现在 Walking + Running Distance(行走+跑步距离)一行(如果已经启用)。此外,依照下面的路径,我们可以看到详细的样本数据:Health Data tab(健康数据选项卡) > Fitness(健身) > Walking + Running Distance(行走+跑步距离) > Show All Data(显示所有数据)。我们的数据就在此列表中。轻击一个单元,我们的图标(目前还未设置)就会与距离一同出现。再次点击此单元,就能看到完整的细节数据。

借助 OneHourWalker,我们便能为全世界 iOS 用户的身体健康贡献一份力量。然而,这只是一个开始。在使用 HealthKit 读取并修改健康数据的道路上,还有非常多的可能性。

当然,对用户而言,拥有这些可追踪数据的好处很多。人们可以轻松地按照日期、星期进行比较,从而激励自己朝着目标努力。不过,真正的伟大之处在于,开发者可以提供全新的,富有创造力的有趣方法来获取数据。

欢迎大家对 HealthKit 应用进行测试。点击此处查看 OneHourWalker 的最终版本。

本文系 OneAPM 工程师编译整理。OneAPM Mobile Insight 以真实用户体验为度量标准进行 Crash 分析,监控网络请求及网络错误,提升用户留存。访问 OneAPM 官方网站感受更多应用性能优化体验,想技术文章,请访问 OneAPM 官方技术博客。

本文转自 OneAPM 官方博客

原文地址:http://www.appcoda.com/healthkit-introduction/

如何借助 HealthKit 打造一款健身应用?相关推荐

  1. 如何从0到1打造一款AI产品?

    2019-11-04 15:56:32 随着AI技术的不断发展,我们看到有越来越多成功的AI产品被研发出来,它们有效地改善了人们的生活与工作.在这样的背景下,AI产品设计正在成为一项重要的技能,被人们 ...

  2. Wecoach陈隽永:如何打造一款完美的智能硬件App?

    入秋的杭州,天气变得和深圳一样深沉,天空忽的下起暴雨. 上周六下午,宅客君在阴沉的背景下和WeCoach创始团队见面的.是的,作为一家创业公司,他们目前还需要周六加一天班来加快进度. 这次拜访,是因为 ...

  3. 如何通过热点打造爆款笔记?6个好写易爆的热点类型

    我们都知道热点话题能够给笔记带来非常多的好处,流量.讨论度等都能够帮助笔记成为爆款,那么今天就来给大家说说如何通过热点打造爆款笔记,以及分享一些好写的热点类型. 一.如何通过热点打造爆款笔记? 核心虽 ...

  4. 如何通过病毒营销打造爆款产品

    病毒营销是通过提供有价值的产品或服务,让用户主动对其进行传播实现"营销杠杆"的作用,逐步获得尽人皆知,尽人参与的成效.如今,病毒式营销已经成为网络营销最为独特的手段,被越来越多的商 ...

  5. Lazada打造爆款秘籍

    新年伊始,万象更新.今年立下的爆单Flag咱一定得实现它!本篇内容小编将献上运营硬货,揭开那些爆款商品和热卖店铺背后的秘密.精细化选品+打造高效引流标题两大攻略,助你2022年销量长虹,日销千单不是梦 ...

  6. 如何打造爆款知识产品?

    知识付费行业的火爆,让越来越多的人关注于"学习".从一开始的不愿付费到现在的愿意对高质量.服务更好的内容付费,显然这样能更加有效地节省用户的信息选择时间和金钱成本,并且这一趋势正在 ...

  7. 实战:深度解析 淘宝打造爆款之淘客推广

    爆款,就是流量大.销量高.人气高.排名高,最重要是转化高了,一个店铺如果有一款爆款,就相当于一个引流利器,如果再把其他产品转化做好,那挣钱真是太容易了.特别是对于很多中小卖家们来说,打造爆款就是求生之 ...

  8. 大数据告诉你:哪款健身运动app最爱?今天你Keep了吗!

    在很多人心目中,运动已经成为一种生活方式,人们的导向也从塑形减肥这类相对具体的目标逐渐过渡成更加宽泛的健康追求.对于追求"精致"的运动爱好者而言,一款好用的健身运动app可以说是不 ...

  9. 抖音人设打造的好处,抖音如何打造爆款视频:国仁楠哥

    随着抖音短视频的兴起越来越多的朋友加入短视频行业,无论是企业还是个人都知道这里隐藏着巨大的利益.当然了遇到最多的问题就是,刚开始做抖音,不知道拍什么内容才能够吸引到更多人的关注,或者说视频爆了,也涨了 ...

最新文章

  1. 用OpenCV4实现图像的超分别率
  2. VR技术怎么落地,行业应用怎么挖掘,来江苏虚拟现实发展大会侃侃
  3. 开源网络备份软件bacula数据恢复(二)
  4. Java中如何锁文件
  5. idea警告Cannot resolve MVC View
  6. STM32——I2C
  7. 54 SD配置-定价配置-分配条件类型到条件排斥组
  8. Highcharts 显示图表
  9. mysql数字辅助表_关于数字的经典SQL编程问题:数字辅助表
  10. 计算机地图制图符号制作的心得,计算机地图制图实习报告.docx
  11. mysql 字段名 减号_MySQL减号
  12. 靶机渗透日记 noob
  13. 进击的WebRTC:我们为什么需要它?
  14. 引用echarts报错Cannot read property ‘init‘ of underfined
  15. lisp语言cond和if套用_方案 – if和cond之间的区别?
  16. html做图片模糊效果,CSS3 filter(滤镜) 制作图片高斯模糊无需JS
  17. storyboard搭建项目_轻松搞定一人一个storyboard开发
  18. centos7安装mysql57--实际操作可行
  19. 什么是IC卡、M1卡,有啥区别
  20. Pycharm更换清华、阿里、豆瓣软件源提高依赖包下载速度

热门文章

  1. less使用语法详解
  2. 移动联通比拼八大热点应用
  3. 易语言-寻找文本和寻找字节集的区别
  4. C语言实现FFT or IFFT
  5. 软件工程专业课实验报告
  6. Vue 添加时间轴 vue-light-timeline
  7. 使用junction为itunes备份目录创建硬链接
  8. 怎么测试t470p性能软件,ThinkPad T470p值得买吗?ThinkPad T470p商务本全面详细评测图解...
  9. 指数增强是什么意思?(附:策略源码)
  10. 安全测试与功能测试的区别是什么?