【发布时间】:2015-01-25 22:03:51
【问题描述】:
我正在开发一个 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];
}
}];
}
【问题讨论】:
-
尝试查询两次。我有一种直觉,需要“启动泵”——第一个查询有点无效。
-
感谢您的建议!但我已经尝试过了,它不起作用。 :-/ 我只是将带有空处理程序的查询放在普通查询之上: [_motionActivityManager queryActivityStartingFromDate: date toDate: [NSDate date] toQueue: _activityQueue withHandler:^(NSArray *activities, NSError *error) {}];
-
也许是队列的问题?我像这样设置队列:_activityQueue.maxConcurrentOperationCount = 1; _activityQueue = [NSOperationQueue 新];
-
我也有同样的问题。不接收最近两分钟的事件。较早的事件正常接收。
-
@pogorskiy 你在这个问题上取得了成功吗?我也面临这个问题,我被困得很严重。