iOS后臺?;?/h2>

iOS后臺?;畎磿r間可分為短時?;?/strong>和長時間?;?/strong>

  • 短時?;?/strong>的方式
    通過beginBackgroundTaskWithName來實現(xiàn)。在iOS7-iOS13可以申請到大約3分鐘的保活時間,在iOS 13以后只能申請30秒左右的時間。

    1. 先通過監(jiān)聽UIApplicationWillEnterForegroundNotification(應(yīng)用進入前臺通知)和UIApplicationDidEnterBackgroundNotification(應(yīng)用進入后臺通知)。
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(appWillEnterForeground) name:UIApplicationWillEnterForegroundNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(appDidEnterBackground) name:UIApplicationDidEnterBackgroundNotification object:nil];
- (void)appWillEnterForeground {}

- (void)appDidEnterBackground {}
  1. 使用Background Task在應(yīng)用進入后臺時開啟?;?,進入前臺時關(guān)閉保活。
@property (assign, nonatomic) UIBackgroundTaskIdentifier backgroundId;
- (void)appWillEnterForeground {
   [self stopKeepAlive];
}

- (void)appDidEnterBackground {
    _backgroundId = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^{
        //申請的時間即將到時回調(diào)該方法
        NSLog(@"BackgroundTask time gone");
        [self stopKeepAlive];
    }];
}

- (void)stopKeepAlive{
  if (_backgroundId) {
        [[UIApplication sharedApplication] endBackgroundTask:_backgroundId];
        _backgroundId = UIBackgroundTaskInvalid;
    }
}

如果想申請多一點時間,可以使用NSTimer循環(huán)申請保活時間,但是建議不要無限申請?;顣r間,因為系統(tǒng)如果發(fā)現(xiàn)該app一直在后臺運行,可能會直接殺掉app。

//開啟定時器 不斷向系統(tǒng)請求后臺任務(wù)執(zhí)行的時間
NSTimer *_timer = [NSTimer scheduledTimerWithTimeInterval:5 target:self selector:@selector(applyForMoreTime) userInfo:nil repeats:YES];
[_timer fire];

//在這里我判斷了申請次數(shù),加上第一次申請?;顣r間的次數(shù)一共6次。
@property(nonatomic,assign) int applyTimes;
-(void)applyForMoreTime {
    if ([UIApplication sharedApplication].backgroundTimeRemaining < 10) {
        _applyTimes += 1;
        NSLog(@"Try to apply for more time:%d",_applyTimes);
        [[UIApplication sharedApplication] endBackgroundTask:_backIden];
        _backIden = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^{
            [self stopKeepAlive];
        }];
        if(_applyTimes == 5){
            [_timer invalidate];
            _applyTimes = 0;
            [self stopKeepAlive];
        }
    }
}
  • 長時間?;?/strong>
    App長時間保活的方式有:播放無聲音樂、后臺持續(xù)定位、后臺下載資源、BGTaskScheduler等,這些需要在蘋果后臺開通后臺權(quán)限,并且在xcode中也開啟相關(guān)權(quán)限。
  1. 播放無聲音樂,適用于音樂類app。像騰訊視頻、愛奇藝等用了播放無聲音樂保活的方式。
    在app進入后臺時開啟無聲音樂,進入前臺后停止無聲音樂。(更好的處理方式是先獲取短時?;?,短時快過時再播放無聲音樂)示例如下:
    監(jiān)聽進入前后臺:
@property (nonatomic, strong) BackgroundPlayer* player;

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(appWillEnterForeground) name:UIApplicationWillEnterForegroundNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(appDidEnterBackground) name:UIApplicationDidEnterBackgroundNotification object:nil];
- (void)appWillEnterForeground {
    if (self.player) {
        [self.player stopPlayBackgroundAlive];
    }
}

- (void)appDidEnterBackground {
    if (_player == nil) {
        _player = [[BackgroundPlayer alloc] init];
    }
    [self.player startPlayer];  
}

編寫音樂播放類:

#import <Foundation/Foundation.h>
#import <AVFoundation/AVFoundation.h>

@interface BackgroundPlayer : NSObject <AVAudioPlayerDelegate>
{
    AVAudioPlayer* _player;
}
- (void)startPlayer;

- (void)stopPlayer;
@end

#import "BackgroundPlayer.h"

@implementation BackgroundPlayer

- (void)startPlayer
{
    if (_player && [_player isPlaying]) {
        return;
    }
    AVAudioSession *session = [AVAudioSession sharedInstance];
    [[AVAudioSession sharedInstance] setMode:AVAudioSessionModeDefault error:nil];

    NSString* route = [[[[[AVAudioSession sharedInstance] currentRoute] outputs] objectAtIndex:0] portType];
    
    if ([route isEqualToString:AVAudioSessionPortHeadphones] || [route isEqualToString:AVAudioSessionPortBluetoothA2DP] || [route isEqualToString:AVAudioSessionPortBluetoothLE] || [route isEqualToString:AVAudioSessionPortBluetoothHFP]) {
        if (@available(iOS 10.0, *)) {
            [[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayAndRecord
                                             withOptions:(AVAudioSessionCategoryOptionMixWithOthers | AVAudioSessionCategoryOptionAllowBluetooth | AVAudioSessionCategoryOptionAllowBluetoothA2DP)
                                                   error:nil];
        } else {
            // Fallback on earlier versions
        }
    }else{
        [[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayAndRecord
                                         withOptions:(AVAudioSessionCategoryOptionMixWithOthers | AVAudioSessionCategoryOptionDefaultToSpeaker)
                                               error:nil];
    }
    
    [session setActive:YES error:nil];
    
    NSURL *url = [[NSBundle bundleWithPath:WECAST_CLOUD_BUNDLE_PATH]URLForResource:@"你的音樂資源" withExtension:nil];
    _player = [[AVAudioPlayer alloc] initWithContentsOfURL:url error:nil];
    [_player prepareToPlay];
    [_player setDelegate:self];
    _player.numberOfLoops = -1;
    BOOL ret = [_player play];
    if (!ret) {
        NSLog(@"play failed,please turn on audio background mode");
    }
}

- (void)stopPlayer
{
    if (_player) {
        [_player stop];
        _player = nil;
        AVAudioSession *session = [AVAudioSession sharedInstance];
        [session setActive:NO error:nil];
        NSLog(@"stop in play background success");
    }
}

@end
  1. 后臺持續(xù)定位

  2. 后臺下載資源
    創(chuàng)建指定標識的后臺NSURLSessionConfiguration,配置好。

    NSURL *url = [NSURL URLWithString:@"https://images.pexels.com/photos/3225517/pexels-photo-3225517.jpeg"];
    NSURLSessionConfiguration *sessionConfig = [NSURLSessionConfiguration backgroundSessionConfigurationWithIdentifier:@"com.qishare.ios.wyw.backgroundDownloadTask"];
// 低于iOS13.0設(shè)備資源下載完后 可以得到通知 AppDelegate.m 文件中的 - (void)application:(UIApplication *)application handleEventsForBackgroundURLSession:(NSString *)identifier completionHandler:(void (^)(void))completionHandler
// iOS13.0+的設(shè)備資源下載完后 直接在下載結(jié)束的代理方法中會有回調(diào)
    sessionConfig.sessionSendsLaunchEvents = YES;
// 當傳輸大數(shù)據(jù)量數(shù)據(jù)的時候,建議將此屬性設(shè)置為YES,這樣系統(tǒng)可以安排對設(shè)備而言最佳的傳輸時間。例如,系統(tǒng)可能會延遲傳輸大文件,直到設(shè)備連接充電器并通過Wi-Fi連接到網(wǎng)絡(luò)為止。 此屬性的默認值為NO。
    sessionConfig.discretionary = YES;
    NSURLSession *session = [NSURLSession sessionWithConfiguration:sessionConfig delegate:self delegateQueue:nil];
    NSURLSessionDownloadTask *downloadTask = [session downloadTaskWithURL:url];
    [downloadTask resume];
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

友情鏈接更多精彩內(nèi)容