核心代码仅需要4句,来实现ios后台保活。
直接上代码
/// 申请进行后台任务
/// 这是一个不间断的申请,在结束的时候再次进行申请
func applyTimeForBackgroundTaskTime() {
var bgTask: UIBackgroundTaskIdentifier!
bgTask = UIApplication.shared.beginBackgroundTask(withName: "bg task") {[weak self] in
UIApplication.shared.endBackgroundTask(bgTask)
self?.applyTimeForBackgroundTaskTime()
}
}
项目设置中,开启后台服务
这里一定要勾选,不然,根本不会进入appdelegate的进入后台的回调中,也就无法启用该函数。
然后,在在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()
}
}
运行结果如下:
风险: 要上架Appstore的话,还是需要说明后台播放音乐的目的的。
完整的demo在github:
https://github.com/gerrywg/WGKeepAliveDemo.git
修订:
在ios15上面好像存在问题,那就还需要播放无声音乐的方式。
更新的代码, 在silentMusic分支, 如下图: