ios - 为什么 CMMotionActivityManager 不返回任何最近的事件数据?

标签 ios iphone

我正在开发一款 iPhone 应用程序,可以跟踪一个人走了多少步以及他们为此投入了多少时间。使用 CMStepCounter 类完成计步。但是,时间戳仅随实时步骤提供,而不会随历史步骤数据一起提供。因此,为了检查步行时实际经过了多少时间,我使用了 CMMotionActivityManager

我有两种方法,一种用于实时计数,一种用于在应用程序处于后台时获取历史数据。实时计数效果很好,但获取历史事件数据的方法却不行!每当应用程序进入后台时,我保存当前日期(通过 NSUserDefaults),当应用程序再次进入前台时,我使用 queryActivityStartingFromDate 将上述日期作为开始日期,并将当前日期作为开始日期结束日期。这样它应该告诉我应用程序暂停期间发生的事件。但通常事件数组会变成空的,即使中间有运动。如果我输入一天的开始作为开始日期,我会得到大量的事件,但有时条目的确切数量会有波动。

我感觉事件记录有延迟。那会是我的问题吗?但是,即使我在恢复应用程序之前等待半小时,也不能保证正确记录中间发生的事件。

获取历史事件数据的方法在本文底部。它在启动时调用,然后每次应用程序返回前台时调用。它总是用

调用
[self calculateHistoricalActivitySince: _enteredBackgroundAt];

_enteredBackground 是一个 NSDate。在启动时,调用一个方法来恢复一些 NSUserDefaults,其中包括这个日期。每次应用程序进入后台时,它都会被设置为当时的日期。当恢复这个日期时,该方法确保它是从同一天开始的——如果它是一个更早的日期(即应用程序最后一次使用是几天前),那么它被设置为当天的开始(今天凌晨 0 点)。 self.activityInSecondsToday 是一个 NSInteger。

-(void)calculateHistoricalActivitySince: (NSDate *) date
{

[_motionActivityManager queryActivityStartingFromDate: date
                                               toDate: [NSDate date]
                                              toQueue: _activityQueue
                                          withHandler:^(NSArray *activities,
                                                        NSError *error)
 {
     if ([error code] == CMErrorUnknown)
     {
         NSLog(@"Motion Activity Manager Error: %@", error);
     }

     // Fill my activity array with historical data if it was walking or running
     else
     {
         NSLog(@"Historical Total Activities from %@ \rto %@:\r %i", date, [NSDate date], (int)[activities count]);

         // Create an array for all relevant activity data
         _activityStorage = [NSMutableArray new];

         for (int i = 0; i < [activities count]; i++)
         {
             if( [[activities objectAtIndex: i] walking] || [[activities objectAtIndex: i] running] || [[activities objectAtIndex: i] unknown])
                 [_activityStorage addObject: [activities objectAtIndex: i]];
         }

         // Calculate the time interval between each entry and increase activityInSecondsToday if the interval is not too large.
         for (int i = 1; i < [_activityStorage count]; i++)
         {
             NSTimeInterval timeInterval = [[[_activityStorage objectAtIndex: i] startDate] timeIntervalSinceDate:[[_activityStorage objectAtIndex: i-1] startDate]];

             if (timeInterval <= _maxTimeBetweenTimestampsForContinuedActivity)
                 self.activityInSecondsToday += timeInterval;

             NSLog(@"ACTIVITY SINCE: %@ \r %ld", date, _activityInSecondsToday);
         }
         NSLog(@"Last activity’s timestamp: %@", [[_activityStorage objectAtIndex: [_activityStorage count]-1] startDate]);

         [_activityStorage removeAllObjects];
     }
 }];

最佳答案

因此,网络上似乎没有太多关于此主题的信息。但我自己想出了一个解决方案,我想与大家分享,以防有人想做类似的事情。

自 iOS 8 起,CMStepCounter 已被弃用。现在有了 CMPedometer,所以我改用了那个 API。 CMMotionActivityManager 也不再是必需的。这使它更容易! CMPedometer 也不会发生上述奇怪的延迟问题。

问题与以前类似:对于实时计数,您会获得步数和时间戳(现在被屏蔽为开始和结束日期,并且所有内容都封装在名为 pedometerData 的容器中);但是对于历史数据,您得到的数据不太精确。新的 API 确实为您提供了历史数据的开始和结束日期,这很棒。但我发现该 API 非常懒惰,因此例如,如果您想要从昨天黎明到昨天午夜的数据,则开始日期将是第一个识别事件的日期和最后一个识别事件的结束日期。中间的一切都将是一项长达数小时的巨大事件!它不会像实时计数那样将事件封装到具有精确时间戳的不同计步器数据对象中。

所以我的解决方案是递归地轮询 API 以获取更小的时间范围。我有一个最近的开始日期,我想要从那时到现在的所有事件信息(步数和步行时间)。所以我选择了我想要涵盖的时间范围,并将其分成更短的时间间隔。然后我递归地询问 [CMPedometer queryPedometerDataFromDate] 每个间隔是否有任何步骤。如果是这样,我将这些步骤添加到我的私有(private)步骤变量中,并将事件时间 (endDate - startDate) 添加到我的私有(private)秒变量中。如果当前间隔还在过去,我将转到下一个间隔并递归地再次做同样的事情。

这样您可以获得相当精确和可靠的结果。当然,你的间隔越小越精确,这个方法执行的次数就越多,耗时也就越长。因此,我有三种不同的间隔大小(一小时、一分钟、五秒),这样可以更快地跳过没有任何事件的时间。显然,优化的潜力很大。

关于ios - 为什么 CMMotionActivityManager 不返回任何最近的事件数据?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28142093/

相关文章:

ios - 如何在TableViewController上给TableView添加约束?

iPhone 动画基于输入值(触摸)而不是时间

iphone - GPS信号丢失警告通知

ios - UIActivityViewController 删除底部

ios - 自动滚动到放置在 UITableViewCell 内的 textView 光标在 iOS11 中不起作用

objective-c - 使用 "Digital heart beat rate sensor (IC)"检测心跳信号 - iOS

ios - 在 iPhone 和 iPad 应用程序之间传输大数据的最佳方式

ios - 应用程序中的状态栏消失了(Xcode iphone)

ios - 为什么tableview的setSelected(_ :animated:) and setHighlighted(_:animated:) method called on init/load?

iphone - iOS - 使用 ImageShack JSON API 上传图片