利用 Healthkit 進(jìn)行睡眠分析(Swift)

原文: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ù)劳翰,然后將其按降序排序。 然后使用startDateendDate以及值的類型(即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項目债鸡,你可以在這里得到

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末铛纬,一起剝皮案震驚了整個濱河市娘锁,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌饺鹃,老刑警劉巖莫秆,帶你破解...
    沈念sama閱讀 221,695評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異悔详,居然都是意外死亡镊屎,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,569評論 3 399
  • 文/潘曉璐 我一進(jìn)店門茄螃,熙熙樓的掌柜王于貴愁眉苦臉地迎上來缝驳,“玉大人,你說我怎么就攤上這事归苍∮糜” “怎么了?”我有些...
    開封第一講書人閱讀 168,130評論 0 360
  • 文/不壞的土叔 我叫張陵拼弃,是天一觀的道長夏伊。 經(jīng)常有香客問我,道長吻氧,這世上最難降的妖魔是什么溺忧? 我笑而不...
    開封第一講書人閱讀 59,648評論 1 297
  • 正文 為了忘掉前任,我火速辦了婚禮盯孙,結(jié)果婚禮上鲁森,老公的妹妹穿的比我還像新娘。我一直安慰自己振惰,他們只是感情好歌溉,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,655評論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著骑晶,像睡著了一般痛垛。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上透罢,一...
    開封第一講書人閱讀 52,268評論 1 309
  • 那天榜晦,我揣著相機(jī)與錄音冠蒋,去河邊找鬼羽圃。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的朽寞。 我是一名探鬼主播识窿,決...
    沈念sama閱讀 40,835評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼脑融!你這毒婦竟也來了喻频?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,740評論 0 276
  • 序言:老撾萬榮一對情侶失蹤肘迎,失蹤者是張志新(化名)和其女友劉穎甥温,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體妓布,經(jīng)...
    沈念sama閱讀 46,286評論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡姻蚓,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,375評論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了匣沼。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片狰挡。...
    茶點(diǎn)故事閱讀 40,505評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖释涛,靈堂內(nèi)的尸體忽然破棺而出加叁,到底是詐尸還是另有隱情,我是刑警寧澤唇撬,帶...
    沈念sama閱讀 36,185評論 5 350
  • 正文 年R本政府宣布它匕,位于F島的核電站,受9級特大地震影響窖认,放射性物質(zhì)發(fā)生泄漏超凳。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,873評論 3 333
  • 文/蒙蒙 一耀态、第九天 我趴在偏房一處隱蔽的房頂上張望轮傍。 院中可真熱鬧,春花似錦首装、人聲如沸创夜。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,357評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽驰吓。三九已至,卻和暖如春系奉,著一層夾襖步出監(jiān)牢的瞬間檬贰,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,466評論 1 272
  • 我被黑心中介騙來泰國打工缺亮, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留翁涤,地道東北人。 一個月前我還...
    沈念sama閱讀 48,921評論 3 376
  • 正文 我出身青樓,卻偏偏與公主長得像葵礼,于是被迫代替她去往敵國和親号阿。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,515評論 2 359

推薦閱讀更多精彩內(nèi)容