原文:http://appcoda.com/sleep-analysis-healthkit/
翻譯:Liberalism
日期:2016年10月5日
現(xiàn)如今仁连,睡眠變革已經(jīng)成為了一種全新的潮流。用戶比以往任何時候都更加關(guān)注自己的睡眠盆均。他們不僅僅關(guān)心自己睡了多久吏垮,同樣也很希望通過一段時間的數(shù)據(jù)收集和分析能夠繪制出他們的睡眠趨勢。而技術(shù)上的進(jìn)步,包括硬件庭再、特別是智能手機(jī)的高速發(fā)展捞奕,使睡眠變革這一高速發(fā)展的領(lǐng)域迎來了全新的曙光。
蘋果在基于安全的前提下拄轻,提供了一種非陈В酷的方式來與用戶的個人健康信息進(jìn)行通信,并通過iOS內(nèi)置的健康
應(yīng)用存儲信息恨搓。作為開發(fā)者不僅可以使用HealyhKit
來打造健康類的App
院促,同時該框架還允許開發(fā)者訪問睡眠數(shù)據(jù),進(jìn)行處理分析奶卓。
在本教程中一疯,針對Healthkit
框架我會帶領(lǐng)大家快速入門,同時會向大家演示如果快速搭建一個簡單的睡眠分析的App
1.簡介
HealthKit
框架結(jié)構(gòu)提供了一個稱之為HealthKit Store
的加密數(shù)據(jù)庫夺姑,開發(fā)者可以使用HKhealth Store這個類來訪問這個數(shù)據(jù)庫墩邀。iPhone和Apple Watch分別有自己的
HealthKit Store,健康數(shù)據(jù)會在iPhone和Apple Watch之間同步盏浙。然而眉睹,Apple Watch為了節(jié)省內(nèi)存空間會自動清理掉一些舊的數(shù)據(jù)。目前
healthKit`框架和健康類的App在iPad上是不支持的废膘。
如果你想創(chuàng)建一個基于健康數(shù)據(jù)的iOS App或者是WatchOS App竹海,HealthKit
框架無疑是非常強(qiáng)大的一個工具。HealthKit設(shè)計的初衷是管理來源廣泛的數(shù)據(jù)丐黄,基于用戶喜好把來源不同的數(shù)據(jù)進(jìn)行自動合并斋配。應(yīng)用程序還可以訪問每個源的原始數(shù)據(jù),并將數(shù)據(jù)本身合并灌闺。App不僅僅用于身體指標(biāo)的檢測艰争、健身或營養(yǎng)情況,還可以用于睡眠分析
那么在接下來的文章里桂对,我會向大家展示在iOS平臺上如何利用HealthKit
框架去存儲甩卓、連接睡眠的分析數(shù)據(jù)。以上的方法也同樣適用于watchOS平臺上應(yīng)用蕉斜。需要注意的是這篇教程使用了Swift2.0和Xcode 7逾柿,所以為了接下來的課程,請確保你目前正在使用的Xcode 7
在我們正式開始之前宅此,請?zhí)崆跋螺d好我們的項目并且解壓机错。我已經(jīng)創(chuàng)建好了基本的UI界面。當(dāng)你運(yùn)行時父腕,你會看到一個計時器的UI界面毡熏,當(dāng)你按下開始按鈕之后,就會發(fā)現(xiàn)開始計時侣诵。
2.使用HealthKit Framework
我們App的目標(biāo)是存儲睡眠的分析信息痢法,并通過開始和結(jié)束兩個按鈕檢索信息狱窘。要使用HealthKit
,首先應(yīng)該在你應(yīng)用的bundle中打開HealthKit
的權(quán)限财搁。在你的項目中蘸炸,在導(dǎo)航中找到當(dāng)前的target -> 再找到 capabilities,然后打開尖奔。
接下來你需要按照以下的代碼在ViewController類里創(chuàng)建一個HKHealthStore的實(shí)例變量
let healthStore = HKHealthStore()
然后搭儒,我們將利用HKHealthStore
這個實(shí)例變量去連接HealthKit Store
這個加密數(shù)據(jù)庫。
如之前所說提茁,HealthKit
允許用戶掌握自己的健康數(shù)據(jù)淹禾,所以在你可以操作、分析用戶的睡眠數(shù)據(jù)之前茴扁,你首先需要去獲取用戶許可铃岔。獲取許可,首先要導(dǎo)入HealthKit Framework峭火,然后如下面一樣更新ViewDidLoad
中的代碼
override func viewDidLoad() {
super.viewDidLoad()
let typestoRead = Set([
HKObjectType.categoryTypeForIdentifier(HKCategoryTypeIdentifierSleepAnalysis)!
])
let typestoShare = Set([
HKObjectType.categoryTypeForIdentifier(HKCategoryTypeIdentifierSleepAnalysis)!
])
self.healthStore.requestAuthorizationToShareTypes(typestoShare, readTypes: typestoRead) { (success, error) -> Void in
if success == false {
NSLog(" Display not allowed")
}
}
}
以上代碼可以提供給用戶同意
或拒絕
的提示毁习,通過block
,你可以在處理成功和失敗后進(jìn)行相應(yīng)的操作并獲得最終的結(jié)果卖丸。沒有必要一直向用戶請求許可纺且,你必須很好的處理程序中的各種錯誤
但是為了避免用戶的誤操作,用戶必須在設(shè)置頁面親自打開允許按鈕稍浆,這樣才能確保真正獲得設(shè)備上健康數(shù)據(jù)的權(quán)限
寫入睡眠分析數(shù)據(jù)
首先载碌,如何去檢索睡眠分析數(shù)據(jù)呢?根據(jù)蘋果官方文檔的說法衅枫,每一個睡眠分析的樣本都有一個唯一值嫁艇,為了確保用戶是躺下并且入睡,HealthKit在同一時間內(nèi)會對兩個或更多的數(shù)據(jù)進(jìn)行采樣为鳄。通過對這些樣本的開始時間和結(jié)束時間進(jìn)行對比裳仆,應(yīng)用程序可以進(jìn)行大量的二次統(tǒng)計和計算腕让。
- 用戶花費(fèi)多少時間入睡孤钦。
- 用戶躺在床上實(shí)際入睡時間所占的比例
- 用戶醒來之后,會在床上躺多久
-
用戶在床上纯丸,以及睡眠時所花費(fèi)的時間匯總
簡明的講偏形,把睡眠分析數(shù)據(jù)儲存到HealthKit store數(shù)據(jù)庫中時,你需要遵循以下方法
- 首先我們需要定義兩個NSDate對象去對應(yīng)開始時間和結(jié)束時間觉鼻。
- 然后我們利用
HKCategoryTypeIdentifierSleepAnalysis
創(chuàng)建一個HKObjectType
的實(shí)例變量
- 我們需要創(chuàng)建一個全新的
HKCategorySample
類型的對象俊扭,通常采用分類樣本的方式來存儲睡眠數(shù)據(jù),獨(dú)立的樣本代表用戶躺在床上或者入睡的時間段坠陈。所以我們可以在同一時間段內(nèi)分別創(chuàng)建出在床上未入睡
以及入睡之后
的樣本
- 最終萨惑,我們就可以利用
HKHealthStore
類中的saveObject
方法把對象存儲起來
編者提示:如果想查看樣本的類型捐康,可以查閱
HealthKit
官方文檔
如果你把以上的注意點(diǎn)和方法轉(zhuǎn)化到Swift中,以下就是把躺床上未入睡
和入睡
的分析數(shù)據(jù)儲存起來的代碼庸蔼,請把以下代碼插入到ViewController
類中
func saveSleepAnalysis() {
// alarmTime and endTime are NSDate objects
if let sleepType = HKObjectType.categoryTypeForIdentifier(HKCategoryTypeIdentifierSleepAnalysis) {
// we create our new object we want to push in Health app
let object = HKCategorySample(type:sleepType, value: HKCategoryValueSleepAnalysis.InBed.rawValue, startDate: self.alarmTime, endDate: self.endTime)
// at the end, we save it
healthStore.saveObject(object, withCompletion: { (success, error) -> Void in
if error != nil {
// something happened
return
}
if success {
print("My new data was saved in HealthKit")
} else {
// something happened again
}
})
let object2 = HKCategorySample(type:sleepType, value: HKCategoryValueSleepAnalysis.Asleep.rawValue, startDate: self.alarmTime, endDate: self.endTime)
healthStore.saveObject(object2, withCompletion: { (success, error) -> Void in
if error != nil {
// something happened
return
}
if success {
print("My new data (2) was saved in HealthKit")
} else {
// something happened again
}
})
}
}
這個方法會在我們想把睡眠分析數(shù)據(jù)存儲到HealthKit
中時被調(diào)用
3.讀取睡眠分析數(shù)據(jù)
- 想要讀取睡眠分析數(shù)據(jù)解总,我們需要創(chuàng)建一個查詢對象。首先需要為
HKCategoryTypeIdentifierSleepAnalysis
定義一個HKObjectType
類型的分類姐仅』ǚ悖或許你希望通過謂詞在開始時間和結(jié)束時間這個你需要的時間段內(nèi)進(jìn)行篩選、檢索數(shù)據(jù)掏膏。你也需要為分類檢索查詢創(chuàng)建 一個分類描述器以獲取我們想要的結(jié)果
您的用于檢索睡眠分析數(shù)據(jù)的代碼應(yīng)如下所示:
func retrieveSleepAnalysis() {
// first, we define the object type we want
if let sleepType = HKObjectType.categoryTypeForIdentifier(HKCategoryTypeIdentifierSleepAnalysis) {
// Use a sortDescriptor to get the recent data first
let sortDescriptor = NSSortDescriptor(key: HKSampleSortIdentifierEndDate, ascending: false)
// we create our query with a block completion to execute
let query = HKSampleQuery(sampleType: sleepType, predicate: nil, limit: 30, sortDescriptors: [sortDescriptor]) { (query, tmpResult, error) -> Void in
if error != nil {
// something happened
return
}
if let result = tmpResult {
// do something with my data
for item in result {
if let sample = item as? HKCategorySample {
let value = (sample.value == HKCategoryValueSleepAnalysis.InBed.rawValue) ? "InBed" : "Asleep"
print("Healthkit sleep: \(sample.startDate) \(sample.endDate) - value: \(value)")
}
}
}
}
// finally, we execute our query
healthStore.executeQuery(query)
}
}
此代碼查詢HealthKit
以獲取所有睡眠分析數(shù)據(jù)劳翰,然后將其按降序排序。 然后使用startDate
和endDate
以及值的類型(即In Bed或Asleep)打印每個查詢馒疹。 我已將限制設(shè)置為30佳簸,以檢索最近30個記錄的樣本。 您還可以使用謂詞方法來選擇自定義的開始和結(jié)束日期行冰。
4.App Testing
對于演示應(yīng)用程序溺蕉,我使用NSTimer
顯示自您按下啟動按鈕以來經(jīng)過的時間。 NSDate
對象在開始和結(jié)束按鈕上創(chuàng)建悼做,以將睡眠分析數(shù)據(jù)保存為已用時間疯特。 在停止操作方法中次企,可以調(diào)用saveSleepAnalysis()
和retrieveSleepAnalysis()
方法來保存和獲取睡眠數(shù)據(jù)拆祈。
@IBAction func stop(sender: AnyObject) {
endTime = NSDate()
saveSleepAnalysis()
retrieveSleepAnalysis()
timer.invalidate()
}
在您的應(yīng)用程序中混驰,您可能需要更改NSDate
對象以選擇相關(guān)的開始和結(jié)束時間(可能不同)椎工,以保存躺在床上的數(shù)據(jù)和睡眠值郭膛。
完成更改后谤牡,您可以運(yùn)行演示應(yīng)用并啟動計時器壶谒。讓它運(yùn)行幾分鐘橱脸,然后點(diǎn)擊停止按鈕葫男。之后打開健康
應(yīng)用程序抱冷。你會發(fā)現(xiàn)睡眠數(shù)據(jù)。
對使用 HealthKit
應(yīng)用的一些建議
HealthKit
旨在為應(yīng)用開發(fā)人員提供一個通用平臺梢褐,以便輕松共享和訪問用戶數(shù)據(jù)旺遮,并避免數(shù)據(jù)中可能的重復(fù)或不一致。蘋果審查指南非常明確的說明應(yīng)用程序使用HealthKit
和訪問用戶讀/寫權(quán)限
必須通過向用戶請求盈咳,但沒有清楚地闡述HealthKit
的使用可能會導(dǎo)致應(yīng)用程序被拒絕耿眉。
將假的或不正確的數(shù)據(jù)保存到健康
的應(yīng)用程序也將被拒絕。 這意味著鱼响,你不能天真地使用算法來計算不同的健康值鸣剪,如本教程中的睡眠分析。 您應(yīng)該嘗試使用內(nèi)置的傳感器數(shù)據(jù)讀取和操作任何參數(shù),以避免計算假數(shù)據(jù)筐骇。
對于完整的Xcode項目债鸡,你可以在這里得到。