在后台持续运行应用程序

时间:2010-10-16 05:27:33

标签: iphone ios cocoa-touch cllocationmanager

我希望我的应用能够继续在后台运行定位服务。为此,我使用了:

-(void)applicationDidEnterBackground:(UIApplication *)application {

    [locationManager stopUpdatingLocation];
    [locationManager startUpdatingLocation];

    //timer=[NSTimer scheduledTimerWithTimeInterval:300 target:self selector:@selector(UpdateLocation) userInfo:nil repeats:YES];

}

但是当我使用NSTimer时,它不会调用UpdateLocation。我尝试使用另一种方法调用它,但之后它也只调用一次。

我希望在后台连续运行应用程序,在定期间隔后检测位置。

3 个答案:

答案 0 :(得分:5)

我在我正在开发的应用程序中这样做了。当应用程序在后台但应用程序不断收到位置更新时,计时器不起作用。我在文档中的某处读过(我现在似乎无法找到它,当我这样做时会发布更新),当应用程序在后台时,只能在活动的运行循环上调用方法。即使在bg中,app委托也有一个活跃的运行循环,所以你不需要创建自己的循环来使这个工作。 [我不确定这是否是正确的解释,但这就是我从我所读到的内容中理解的]

首先,在应用的info.plist中添加“背景模式”键的“位置”对象。现在,您需要做的是在应用中的任何位置启动位置更新:

CLLocationManager locationManager = [[CLLocationManager alloc] init];

locationManager.delegate = self;//or whatever class you have for managing location

[locationManager startUpdatingLocation];

接下来,编写一个方法来处理位置更新, 在应用程序委托中说 - (void)didUpdateToLocation:(CLLocation *)位置。然后在启动位置管理器的类中实现方法locationManager:didUpdateLocation:fromLocation of CLLocationManagerDelegate(因为我们将位置管理器委托设置为'self')。在此方法中,您需要检查是否已经过了必须处理位置更新的时间间隔。您可以通过每次保存当前时间来完成此操作。如果该时间已过,请从您的app delegate调用方法UpdateLocation:

NSDate *newLocationTimestamp = newLocation.timestamp;
NSDate *lastLocationUpdateTiemstamp;

int locationUpdateInterval = 300;//5 mins

NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
if (userDefaults) {

        lastLocationUpdateTiemstamp = [userDefaults objectForKey:kLastLocationUpdateTimestamp];

        if (!([newLocationTimestamp timeIntervalSinceDate:lastLocationUpdateTiemstamp] < locationUpdateInterval)) {
            //NSLog(@"New Location: %@", newLocation);
            [(AppDelegate*)[UIApplication sharedApplication].delegate didUpdateToLocation:newLocation];
            [userDefaults setObject:newLocationTimestamp forKey:kLastLocationUpdateTimestamp];
        }
    }
}

即使您的应用处于后台,这也会每隔5分钟调用您的方法。 Imp:此实现耗尽电池,如果您的位置数据的准确性不重要,您应该使用[locationManager startMonitoringSignificantLocationChanges]

在将此添加到您的应用之前,请阅读位置感知编程指南 http://developer.apple.com/library/ios/#documentation/UserExperience/Conceptual/LocationAwarenessPG/Introduction/Introduction.html

答案 1 :(得分:0)

我知道答案已经很晚了。但是,如果你还没有得到答案,这就是我所做的,我的应用程序在后台持续运行。

- (void)applicationDidEnterBackground:(UIApplication *)application
{
     UIApplication *app = [UIApplication sharedApplication];
     UIBackgroundTaskIdentifier bgTask = 0;

     backgroundTimer = [NSTimer scheduledTimerWithTimeInterval:10 target:self  selector:@selector(backgroundTask) userInfo:nil repeats:YES];

     bgTask = [app beginBackgroundTaskWithExpirationHandler:^{
        [app endBackgroundTask:bgTask];
    }];
}

现在在 backgroundTask 方法中执行任何操作。

答案 2 :(得分:0)

您可以在后台运行应用程序以执行特定任务。

- (void)applicationDidEnterBackground:(UIApplication *)application
{
    UIApplication *app = [UIApplication sharedApplication];
    UIBackgroundTaskIdentifier bgTask = 0;

    backgroundTimer = [NSTimer scheduledTimerWithTimeInterval:10 target:self selector:@selector(backgroundTask) userInfo:nil repeats:YES];

    bgTask = [app beginBackgroundTaskWithExpirationHandler:^{
        [app endBackgroundTask:bgTask];
    }];
}