NSTimer在后台运行

在iOS中应用进入后台,系统会将整个应用当前的状态以快照的形式保存起来,也就是常说的"墓碑"机制。此时对应的定时器也就直接被关闭了,如果我们需要在后台保持一段时间运行,我们就需要做对应的处理。
在iOS中应用进入后台,app都有3分钟左右的后台任务执行时间。 3分钟后,app会被iOS强行挂起。

除了

  • 音乐应用
  • 电话之类
  • GPS定位

的app可以保持一个长久的后台运行。当你提交app到App Store时,苹果会审查你的app,一旦发现你“滥用”了后台API,你的app将被拒绝。也即是说你在info.plist 设置这几种backgroundmode,你的程序必须含有这些功能,你的程序才会有审核通过。

一般情况下,3分钟的后台时间足够我们处理我们的应用数据了。
定时器在这3分钟的时间内保持不间断的运行。

方法1

在app的代理中添加如下代码

- (void)applicationDidEnterBackground:(UIApplication *)application {
    if ( [self isMutiltaskingSupported] == NO) {
        NSLog(@"---> 不支持后台多任务");
        return;
    }
    
    UIApplication*   app = [UIApplication sharedApplication];
    __block    UIBackgroundTaskIdentifier bgTask;
    bgTask = [app beginBackgroundTaskWithExpirationHandler:^{
        dispatch_async(dispatch_get_main_queue(), ^{
            if (bgTask != UIBackgroundTaskInvalid)
            {
                bgTask = UIBackgroundTaskInvalid;
            }
        });
    }];
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        dispatch_async(dispatch_get_main_queue(), ^{
            if (bgTask != UIBackgroundTaskInvalid)
            {
                bgTask = UIBackgroundTaskInvalid;
            }
        });
    });
}

当然你需要判断你的设备是否支持后台(不支持多任务情况现在基本可以忽略)

// 判断当前设备是否支持 后台多任务
- (BOOL)isMutiltaskingSupported{
    BOOL result = NO;
    if ( [[UIDevice currentDevice] respondsToSelector:@selector(isMultitaskingSupported)]) {
        result = [[UIDevice currentDevice] isMultitaskingSupported];
    }
    return result;
}
方法2

在网上搜索了下,基本上实现方式都差不多。这里直接代码贴出,以备以后查阅

#import "AppDelegate.h"

@interface AppDelegate ()
@property (nonatomic, unsafe_unretained) UIBackgroundTaskIdentifier backgroundTaskIdentifier;
@property (nonatomic, strong) NSTimer * myTimer;

@end

@implementation AppDelegate

// 判断当前设备是否支持 后台多任务
- (BOOL)isMutiltaskingSupported{
    BOOL result = NO;
    if ( [[UIDevice currentDevice] respondsToSelector:@selector(isMultitaskingSupported)]) {
        result = [[UIDevice currentDevice] isMultitaskingSupported];
    }
    return result;
}
// 定时器调用方法
- (void)timerMethod:(NSTimer *)paramSender{
    NSTimeInterval backgroundTimeRemanging = [[UIApplication sharedApplication] backgroundTimeRemaining];
    if ( backgroundTimeRemanging == DBL_MAX) {
        NSLog(@"Background Time Remaining = Undeterminded");
    }
    //--显示后台任务还剩余的时间
    NSLog(@"Background Timer Remaining = %.02f Seconds", backgroundTimeRemanging);
}

- (void)applicationDidEnterBackground:(UIApplication *)application {
    NSTimeInterval backgroundTimeRemanging = [[UIApplication sharedApplication] backgroundTimeRemaining];
    NSLog(@"backgroundTimeRemanging = %.02f", backgroundTimeRemanging);
    
    if ( [self isMutiltaskingSupported] == NO) {
        NSLog(@"---> 不支持后台多任务");
        return;
    }
    
    self.backgroundTaskIdentifier = [application beginBackgroundTaskWithExpirationHandler:^{
        [self endBackgroundTask];
    }];
    self.myTimer = [NSTimer scheduledTimerWithTimeInterval:1.0f target:self selector:@selector(timerMethod:) userInfo:nil repeats:YES];
}

// 任务完成,处理释放对象
- (void)endBackgroundTask{
    dispatch_queue_t mainQueue = dispatch_get_main_queue();
    __weak AppDelegate *weakSelf = self;
    dispatch_async(mainQueue, ^{
        AppDelegate * strongSelf = weakSelf;
        if (strongSelf != nil) {
            [strongSelf.myTimer invalidate];
            [[UIApplication sharedApplication] endBackgroundTask:self.backgroundTaskIdentifier];
            strongSelf.backgroundTaskIdentifier = UIBackgroundTaskInvalid;
        }
    });
}

// 进入前台,停止任务
- (void)applicationDidBecomeActive:(UIApplication *)application {
    if (self.backgroundTaskIdentifier != UIBackgroundTaskInvalid){
        [self endBackgroundTask];
    }
}
@end

参考链接
参考1

你可能感兴趣的:(NSTimer在后台运行)