【问题标题】:Why does CMMotionActivityManager not return any recent activity data?为什么 CMMotionActivityManager 不返回任何最近的活动数据?
【发布时间】: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 你在这个问题上取得了成功吗?我也面临这个问题,我被困得很严重。

标签: ios iphone


【解决方案1】:

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

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

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

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

这样您可以获得相当精确和可靠的结果。当然,您的间隔越小、越精确,此方法执行的频率越高,所需的时间就越长。因此,我有三种不同的间隔大小(一小时、一分钟、5 秒),这样可以更快地跳过没有任何活动的时间。显然,优化的潜力很大。

【讨论】:

    【解决方案2】:

    我自己的错误也有同样的问题,即 fromDate 比现在晚。通过将值从 7 更改为 -7,它可以按预期工作。

    // in Swift 3
    let day_7daysAgo = NSCalendar.current.date(byAdding:  
        Calendar.Component.day, value: -7, to: Date())!
    
    motActMgr.queryActivityStarting(from: day_7daysAgo, to: Date(), to: OperationQueue.main) { (ma_: [CMMotionActivity]?, e: Error?) in
    }
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2016-08-13
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2015-08-08
      相关资源
      最近更新 更多