iOS开发-读取“健康”中的步数和步行+跑步距离

2023-11-22 15:10

本文主要是介绍iOS开发-读取“健康”中的步数和步行+跑步距离,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

借鉴于
http://www.csdn.net/article/2015-01-23/2823686-healthkit-tutorial-with-swift/4

  • 注:iOS10遇到的错误:
    Terminating app due to uncaught exception 'NSInvalidArgumentException',
    reason: 'NSHealthUpdateUsageDescription must be set in the app's Info.plist
    in order to request write authorization.'
    解决办法:
    我们要在info.plist文件中声明苹果健康的使用权限,所以在info.plist中添加以下key就可以了。请求写入和请求读取都需要添加!
    请求写入
    <key>NSHealthUpdateUsageDescription</key>
    <string>some string value stating the reason</string>
    请求读取
    <key>NSHealthShareUsageDescription</key><string>some string value stating the reason</string>
    从iOS 10 因苹果健康导致闪退 crash - 借鉴与简书作者文章 找到的解决办法,谢谢

1、第一步首先需要开启HealthKit


HealthKit.png

2、新建一个HealthKitManage类,继承于NSObject

  • (1)首先在.h文件里面声明一个属性
@property (nonatomic, strong) HKHealthStore *healthStore;
  • (2)导入头文件
#import <HealthKit/HealthKit.h>
#import <UIKit/UIDevice.h>#define HKVersion [[[UIDevice currentDevice] systemVersion] doubleValue]
#define CustomHealthErrorDomain @"com.sdqt.healthError"
  • (3)创建单例方法
+(id)shareInstance
{static id manager ;static dispatch_once_t onceToken;dispatch_once(&onceToken, ^{manager = [[[self class] alloc] init];});return manager;
}
  • (4)检查是否支持获取健康数据
/**  @brief  检查是否支持获取健康数据*/
- (void)authorizeHealthKit:(void(^)(BOOL success, NSError *error))compltion
{if(HKVersion >= 8.0){if (![HKHealthStore isHealthDataAvailable]) {NSError *error = [NSError errorWithDomain: @"com.raywenderlich.tutorials.healthkit" code: 2 userInfo: [NSDictionary dictionaryWithObject:@"HealthKit is not available in th is Device"                                                                      forKey:NSLocalizedDescriptionKey]];if (compltion != nil) {compltion(false, error);}return;}if ([HKHealthStore isHealthDataAvailable]) {if(self.healthStore == nil)self.healthStore = [[HKHealthStore alloc] init];/*组装需要读写的数据类型*/NSSet *writeDataTypes = [self dataTypesToWrite];NSSet *readDataTypes = [self dataTypesRead];/*注册需要读写的数据类型,也可以在“健康”APP中重新修改*/[self.healthStore requestAuthorizationToShareTypes:writeDataTypes readTypes:readDataTypes completion:^(BOOL success, NSError *error) {if (compltion != nil) {NSLog(@"error->%@", error.localizedDescription);compltion (success, error);}}];}}else {NSDictionary *userInfo = [NSDictionary dictionaryWithObject:@"iOS 系统低于8.0"                                                                      forKey:NSLocalizedDescriptionKey];NSError *aError = [NSError errorWithDomain:CustomHealthErrorDomain code:0 userInfo:userInfo];compltion(0,aError);}
}/*!*  @brief  写权限*  @return 集合*/
- (NSSet *)dataTypesToWrite
{HKQuantityType *heightType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierHeight];HKQuantityType *weightType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierBodyMass];HKQuantityType *temperatureType = [HKQuantityType quantityTypeForIdentifier:HKQuantityTypeIdentifierBodyTemperature];HKQuantityType *activeEnergyType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierActiveEnergyBurned];return [NSSet setWithObjects:heightType, temperatureType, weightType,activeEnergyType,nil];
}/*!*  @brief  读权限*  @return 集合*/
- (NSSet *)dataTypesRead
{HKQuantityType *heightType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierHeight];HKQuantityType *weightType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierBodyMass];HKQuantityType *temperatureType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierBodyTemperature];HKCharacteristicType *birthdayType = [HKObjectType characteristicTypeForIdentifier:HKCharacteristicTypeIdentifierDateOfBirth];HKCharacteristicType *sexType = [HKObjectType characteristicTypeForIdentifier:HKCharacteristicTypeIdentifierBiologicalSex];HKQuantityType *stepCountType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierStepCount];HKQuantityType *distance = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierDistanceWalkingRunning];HKQuantityType *activeEnergyType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierActiveEnergyBurned];return [NSSet setWithObjects:heightType, temperatureType,birthdayType,sexType,weightType,stepCountType, distance, activeEnergyType,nil];
}
  • (5)读取步数
//获取步数
- (void)getStepCount:(void(^)(double value, NSError *error))completion
{HKQuantityType *stepType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierStepCount];NSSortDescriptor *timeSortDescriptor = [[NSSortDescriptor alloc] initWithKey:HKSampleSortIdentifierEndDate ascending:NO];// Since we are interested in retrieving the user's latest sample, we sort the samples in descending order, and set the limit to 1. We are not filtering the data, and so the predicate is set to nil.HKSampleQuery *query = [[HKSampleQuery alloc] initWithSampleType:stepType predicate:[HealthKitManage predicateForSamplesToday] limit:HKObjectQueryNoLimit sortDescriptors:@[timeSortDescriptor] resultsHandler:^(HKSampleQuery *query, NSArray *results, NSError *error) {if(error){completion(0,error);}else{NSInteger totleSteps = 0;for(HKQuantitySample *quantitySample in results){HKQuantity *quantity = quantitySample.quantity;HKUnit *heightUnit = [HKUnit countUnit];double usersHeight = [quantity doubleValueForUnit:heightUnit];totleSteps += usersHeight;}NSLog(@"当天行走步数 = %ld",(long)totleSteps);completion(totleSteps,error);}}];[self.healthStore executeQuery:query];
}
  • (6)读取步行+跑步距离
//获取公里数
- (void)getDistance:(void(^)(double value, NSError *error))completion
{HKQuantityType *distanceType = [HKObjectType quantityTypeForIdentifier:HKQuantityTypeIdentifierDistanceWalkingRunning];NSSortDescriptor *timeSortDescriptor = [[NSSortDescriptor alloc] initWithKey:HKSampleSortIdentifierEndDate ascending:NO];HKSampleQuery *query = [[HKSampleQuery alloc] initWithSampleType:distanceType predicate:[HealthKitManage predicateForSamplesToday] limit:HKObjectQueryNoLimit sortDescriptors:@[timeSortDescriptor] resultsHandler:^(HKSampleQuery * _Nonnull query, NSArray<__kindof HKSample *> * _Nullable results, NSError * _Nullable error) {if(error){completion(0,error);}else{double totleSteps = 0;for(HKQuantitySample *quantitySample in results){HKQuantity *quantity = quantitySample.quantity;HKUnit *distanceUnit = [HKUnit meterUnitWithMetricPrefix:HKMetricPrefixKilo];double usersHeight = [quantity doubleValueForUnit:distanceUnit];totleSteps += usersHeight;}NSLog(@"当天行走距离 = %.2f",totleSteps);completion(totleSteps,error);}}];[self.healthStore executeQuery:query];
}
  • (7)、NSPredicate当天时间段的方法实现
/*!*  @brief  当天时间段**  @return 时间段*/
+ (NSPredicate *)predicateForSamplesToday {NSCalendar *calendar = [NSCalendar currentCalendar];NSDate *now = [NSDate date];NSDateComponents *components = [calendar components:NSCalendarUnitYear|NSCalendarUnitMonth|NSCalendarUnitDay fromDate:now];[components setHour:0];[components setMinute:0];[components setSecond: 0];NSDate *startDate = [calendar dateFromComponents:components];NSDate *endDate = [calendar dateByAddingUnit:NSCalendarUnitDay value:1 toDate:startDate options:0];NSPredicate *predicate = [HKQuery predicateForSamplesWithStartDate:startDate endDate:endDate options:HKQueryOptionNone];return predicate;
}

3、在控制器里展示出来读取的数据
(1)、首先导入头文件,并添加Label

#import "HealthKitManage.h"
@interface ViewController ()@end@implementation ViewController
{UILabel *stepLabel;UILabel *distanceLabel;
}

(2)创建界面,展示数据

- (void)viewDidLoad {[super viewDidLoad];// Do any additional setup after loading the view, typically from a nib.UIButton *btn1 = [UIButton buttonWithType:UIButtonTypeCustom];btn1.frame = CGRectMake(50, 100, 100, 40);[btn1 setTitle:@"计步" forState:UIControlStateNormal];btn1.backgroundColor = [UIColor cyanColor];[self.view addSubview:btn1];[btn1 addTarget:self action:@selector(onClickBtn1) forControlEvents:UIControlEventTouchUpInside];UIButton *btn2 = [UIButton buttonWithType:UIButtonTypeCustom];btn2.frame = CGRectMake(50, 160, 100, 40);[btn2 setTitle:@"距离" forState:UIControlStateNormal];btn2.backgroundColor = [UIColor cyanColor];[self.view addSubview:btn2];[btn2 addTarget:self action:@selector(onClickBtn2) forControlEvents:UIControlEventTouchUpInside];stepLabel = [[UILabel alloc] initWithFrame:CGRectMake(50, 220, 200, 40)];stepLabel.backgroundColor = [UIColor cyanColor];[self.view addSubview:stepLabel];distanceLabel = [[UILabel alloc] initWithFrame:CGRectMake(50, 280, 200, 40)];distanceLabel.backgroundColor = [UIColor cyanColor];[self.view addSubview:distanceLabel];
}- (void)onClickBtn1
{HealthKitManage *manage = [HealthKitManage shareInstance];[manage authorizeHealthKit:^(BOOL success, NSError *error) {if (success) {NSLog(@"success");[manage getStepCount:^(double value, NSError *error) {NSLog(@"1count-->%.0f", value);NSLog(@"1error-->%@", error.localizedDescription);dispatch_async(dispatch_get_main_queue(), ^{stepLabel.text = [NSString stringWithFormat:@"步数:%.0f步", value];});}];}else {NSLog(@"fail");}}];
}- (void)onClickBtn2
{HealthKitManage *manage = [HealthKitManage shareInstance];[manage authorizeHealthKit:^(BOOL success, NSError *error) {if (success) {NSLog(@"success");[manage getDistance:^(double value, NSError *error) {NSLog(@"2count-->%.2f", value);NSLog(@"2error-->%@", error.localizedDescription);dispatch_async(dispatch_get_main_queue(), ^{distanceLabel.text = [NSString stringWithFormat:@"公里数:%.2f公里", value];});}];}else {NSLog(@"fail");}}];
}

4、展示成果

这篇关于iOS开发-读取“健康”中的步数和步行+跑步距离的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/411010

相关文章

Python通过模块化开发优化代码的技巧分享

《Python通过模块化开发优化代码的技巧分享》模块化开发就是把代码拆成一个个“零件”,该封装封装,该拆分拆分,下面小编就来和大家简单聊聊python如何用模块化开发进行代码优化吧... 目录什么是模块化开发如何拆分代码改进版:拆分成模块让模块更强大:使用 __init__.py你一定会遇到的问题模www.

Spring Security基于数据库的ABAC属性权限模型实战开发教程

《SpringSecurity基于数据库的ABAC属性权限模型实战开发教程》:本文主要介绍SpringSecurity基于数据库的ABAC属性权限模型实战开发教程,本文给大家介绍的非常详细,对大... 目录1. 前言2. 权限决策依据RBACABAC综合对比3. 数据库表结构说明4. 实战开始5. MyBA

使用Python开发一个简单的本地图片服务器

《使用Python开发一个简单的本地图片服务器》本文介绍了如何结合wxPython构建的图形用户界面GUI和Python内建的Web服务器功能,在本地网络中搭建一个私人的,即开即用的网页相册,文中的示... 目录项目目标核心技术栈代码深度解析完整代码工作流程主要功能与优势潜在改进与思考运行结果总结你是否曾经

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

利用Python开发Markdown表格结构转换为Excel工具

《利用Python开发Markdown表格结构转换为Excel工具》在数据管理和文档编写过程中,我们经常使用Markdown来记录表格数据,但它没有Excel使用方便,所以本文将使用Python编写一... 目录1.完整代码2. 项目概述3. 代码解析3.1 依赖库3.2 GUI 设计3.3 解析 Mark

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

解决Java中基于GeoTools的Shapefile读取乱码的问题

《解决Java中基于GeoTools的Shapefile读取乱码的问题》本文主要讨论了在使用Java编程语言进行地理信息数据解析时遇到的Shapefile属性信息乱码问题,以及根据不同的编码设置进行属... 目录前言1、Shapefile属性字段编码的情况:一、Shp文件常见的字符集编码1、System编码

利用Python实现添加或读取Excel公式

《利用Python实现添加或读取Excel公式》Excel公式是数据处理的核心工具,从简单的加减运算到复杂的逻辑判断,掌握基础语法是高效工作的起点,下面我们就来看看如何使用Python进行Excel公... 目录python Excel 库安装Python 在 Excel 中添加公式/函数Python 读取

基于Python开发批量提取Excel图片的小工具

《基于Python开发批量提取Excel图片的小工具》这篇文章主要为大家详细介绍了如何使用Python中的openpyxl库开发一个小工具,可以实现批量提取Excel图片,有需要的小伙伴可以参考一下... 目前有一个需求,就是批量读取当前目录下所有文件夹里的Excel文件,去获取出Excel文件中的图片,并