向应用程序发送静默推送通知,更新位置并在后台发送到服务器

Yos*_*far 5 objective-c push-notification ios background-task background-fetch

我想向后台应用程序发送静默推送通知,然后获取当前用户位置并将其发送到 Web 服务。

我实现了推送通知方法以及这两个方法:

- (void)application:(UIApplication *)application performFetchWithCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
{
    NSDate *fetchStart = [NSDate date];

    [self sendLocationToServerWithCompletionHandler:^(UIBackgroundFetchResult result) {
        completionHandler(result);

        NSDate *fetchEnd = [NSDate date];
        NSTimeInterval timeElapsed = [fetchEnd timeIntervalSinceDate:fetchStart];
        NSLog(@"Background Fetch Duration: %f seconds", timeElapsed);

    }];
}

- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
{

}
Run Code Online (Sandbox Code Playgroud)

我还创建了一个将位置发送到服务器的方法:

- (void)sendLocationToServerWithCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
{
    NSDictionary *params = @{
                             @"UserId"      : self.userId,
                             @"Latitude"    : self.latitude,
                             @"Longitude"   : self.longitude
                             }

    ServerManager *manager = [ServerManager sharedManager];
    [manager sendLocationToServerWithCompletion:^(BOOL success) {

        if (success)
        {
            completionHandler(UIBackgroundFetchResultNewData);
        }
        else
        {
            completionHandler(UIBackgroundFetchResultFailed);
        }

    }];
}
Run Code Online (Sandbox Code Playgroud)

我只是无法理解它们是如何协同工作的,Apple 是否会批准这一点,是否有可能以及位置背景获取进入何处。

提前致谢。

Gru*_*kes 2

以下是您可以采取哪些措施的简要概述,以便为您提供一个想法。假设有一个作为单例实现的模型类,并且有一些伪代码。

// App delegate
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
{
    completionHandler(UIBackgroundFetchResultNewData);
    [[YourModel singleton] pushNotificationReceived: userInfo];
}

// Model
- (void)  pushNotificationReceived:(NSDictionary *) userInfo
{
    [self registerBackgroundTaskHandler];
    get the location here, or start getting the location
    [self sendLocationToServerWithCompletionHandler: your completion handler];
}


- (void) registerBackgroundTaskHandler
{
    __block UIApplication *app = [UIApplication sharedApplication];
    self.backgroundTaskId = [app beginBackgroundTaskWithExpirationHandler:^{
        DDLogInfo(@"BACKGROUND  Background task expiration handler called");
        [app endBackgroundTask:self.backgroundTaskId];
        self.backgroundTaskId = 0;
    }];
}


- (void) endBackgroundTask
{
    if (self.backgroundTaskId)
    {
        UIApplication *app = [UIApplication sharedApplication];
        [app endBackgroundTask:self.backgroundTaskId];
        self.backgroundTaskId = 0;
    }
}
Run Code Online (Sandbox Code Playgroud)

您需要先获取位置,然后才能发送。如果您只获取一个位置并且使用的是 iOS9,则可以使用 CLLocationManager:requestLocation: 并且您可以相对轻松地将其放入我所说的“在此处获取位置”中。

如果您不使用 iOS 9(requestLocation 是 iOS 9 中的新功能),那么它会更复杂一些。如何使用位置管理器本身就是一个主题,这里发布的代码太多。您需要阅读和研究有关使用位置管理器的所有内容,然后才能合并它。如果您需要位置更新流,它会变得更加复杂,并且其中所说的“或开始获取位置”比伪代码中隐含的内容要复杂得多。

我的建议是,从 iOS9 开始,获取该位置的一个实例,然后在工作正常时,根据需要添加更多功能或 iOS8 支持。