ios后台保活-极简方式

核心代码仅需要4句,来实现ios后台保活。
直接上代码

    /// 申请进行后台任务
    /// 这是一个不间断的申请,在结束的时候再次进行申请
    func applyTimeForBackgroundTaskTime() {
        var bgTask: UIBackgroundTaskIdentifier!
        bgTask = UIApplication.shared.beginBackgroundTask(withName: "bg task") {[weak self] in
            UIApplication.shared.endBackgroundTask(bgTask)
            self?.applyTimeForBackgroundTaskTime()
        }
    }

项目设置中,开启后台服务
这里一定要勾选,不然,根本不会进入appdelegate的进入后台的回调中,也就无法启用该函数。

image-20211004113048121.png

然后,在在appdelegate即将进入后台的时候,调用这个函数,即可实现后台保活。

原理解析:

上面的函数applyTimeForBackgroundTaskTime中, 实际就是申请时间执行一段后台任务,并返回一个后台任务id。因为该时间是有限制的,一般30秒。该申请的任务结束回调中,必须要结束该后台任务。不然系统就会杀死该App。在这个结束的回调中,再次申请执行一段后台任务,则系统又会开始新的申请,并不会真正的挂起。

注意:

这里可能有同学会有疑问,结束后台任务和杀死App有啥不同?说来还真是不同的,一个App进入后台,系统会将其挂起suspend,杀死是end。挂起的意思是不运行,进入前台仍然从进入后台的时刻开始运行。而杀死后,如果切到该程序,则会重新启动。 结束后台任务后,系统仅会将App挂起,而不会杀死。但如果不在结束任务的回调中进行结束任务,则系统就会杀死。被挂起的程序,在系统资源不够的时候也会被杀死,那是另外的事情,不在本文的讨论范围。

这样做的好处
省去了复杂的后台播放音乐的逻辑,也不需要导入AVFoundation框架。
代码逻辑非常简化.
完整的示例代码如下

//
//  AppBackgroundTaskManager.swift
//  BackgroundTask
//
//  Created by wanggang on 2021/9/25.
//

import UIKit

class AppBackgroundTaskManager: NSObject {
    static let shared = AppBackgroundTaskManager.init()
    
    func startToApplyTimeForBackgroundTask() {
        applyTimeForBackgroundTaskTime()
    }
    
    /// 申请进行后台任务
    /// 这是一个不间断的申请,在结束的时候再次进行申请
    func applyTimeForBackgroundTaskTime() {
        var bgTask: UIBackgroundTaskIdentifier!
        bgTask = UIApplication.shared.beginBackgroundTask(withName: "bg task") {[weak self] in
            UIApplication.shared.endBackgroundTask(bgTask)
            self?.applyTimeForBackgroundTaskTime()
        }
    }
    
}

AppDelegate中的调用, 以及验证代码如下.

//
//  AppDelegate.swift
//  BackgroundTask
//
//  Created by wanggang on 2021/9/25.
//

import UIKit

@main
class AppDelegate: UIResponder, UIApplicationDelegate {
    var taskTimer: Timer?

    var window: UIWindow?

    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
        // Override point for customization after application launch.
        return true
    }
    
    func applicationDidEnterBackground(_ application: UIApplication) {
        // 调用申请后台保活
        AppBackgroundTaskManager.shared.startToApplyTimeForBackgroundTask()
        
        // 测试代码,打印看一下后台保活或者模拟后台干一些其他事情。
        // 说明: 后台保活就是上面的AppBackgroundTaskManager.shared.startApplyBackgroundTask()
        // 而在后台干事情,不一定非要这个位置写代码。可以在项目的任何位置,如在某个页面发起一些网络请求。
        doSomethingInBackground()
    }
    
    func doSomethingInBackground() {
        // 后台做一些事情
        taskTimer = Timer.scheduledTimer(withTimeInterval: 5, repeats: true) { _ in
            print("doing some thing:\(UIApplication.shared.backgroundTimeRemaining)")
        }
    }
    
    func stopDoingSomethingInBackgroud() -> () {
        taskTimer?.invalidate()
        taskTimer = nil
    }
    
    func applicationWillEnterForeground(_ application: UIApplication) {
        stopDoingSomethingInBackgroud()
    }


}

运行结果如下:

image-20211004114654886.png

风险: 要上架Appstore的话,还是需要说明后台播放音乐的目的的。

完整的demo在github:
https://github.com/gerrywg/WGKeepAliveDemo.git

修订:
在ios15上面好像存在问题,那就还需要播放无声音乐的方式。
更新的代码, 在silentMusic分支, 如下图:

image.png

你可能感兴趣的:(ios后台保活-极简方式)