我正在尝试使用applicationWillResignActive()
以便在应用程序进入后台之前将一些数据同步到我的Firestore
数据库。
func applicationWillResignActive(_ application: UIApplication) {
self.uploadWantToPlay()
}
当我从applicationWillResignActive()
调用我的上传函数时,它会运行,但在下次应用程序变为活动状态之前,不会向Firestore
添加任何数据。
当我出于测试目的从我的一个ViewControllers
运行相同的功能时,数据会立即添加到Firestore
。
我也尝试从applicationDidEnterBackground()
调用该函数,我尝试在它自己的DispatchQueue
中运行它。但它也有同样的结果。
如何在用户即将离开应用并使其正确执行数据库同步时运行此功能?
处理数据库同步的函数;
func uploadWantToPlay() {
print ("Inside uploadWantToPlay")
if let wantToPlay = User.active.wantToPlayList {
if let listEntries = wantToPlay.list_entries {
let cleanedEntries = listEntries.compactMap({ (entry: ListEntry) -> ListEntry? in
if entry.game?.first_release_date != nil {
return entry
} else {
return nil
}
})
let gamesToUpload = cleanedEntries.filter {
$0.game!.first_release_date! > Int64(NSDate().timeIntervalSince1970 * 1000)
}
DatabaseConnection().writeWantToPlayToDatabase(user: User.active,wantToPlay: gamesToUpload)
}
}
}
func writeWantToPlayToDatabase(user: User, wantToPlay: [ListEntry]) {
firebaseSignIn()
let deviceId = ["(user.deviceId)": "Device ID"]
for entry in wantToPlay {
let wantToPlayGameRef = fireStore.collection(WANTTOPLAY).document("(entry.game!.id!)")
wantToPlayGameRef.updateData(deviceId) {(err) in
if err != nil {
wantToPlayGameRef.setData(deviceId) {(err) in
if let err = err {
Events().logException(withError: err, withMsg: "DatabaseConnection-writeWantToPlayToDatabase(user, [ListEntry]) Failed to write to database")
} else {
print("Document successfully written to WantToPlayGames")
}
}
} else {
print("Document successfully updated in WantToPlayGames")
}
}
}
}
根据苹果文档
移动到后台的应用程序预计将将自己置于 尽快处于静止状态,以便可以挂起它们 由系统。如果你的应用正在执行任务,并且需要 完成该任务的额外时间很少,它可以调用 beginBackgroundTaskWithName:expirationHandler: or beginBackgroundTaskWithExfilationHandler: UI APIpplication 的方法 对象以请求一些额外的执行时间。调用以下任一 这些方法会暂时延迟应用的挂起,从而 一点额外的时间来完成它的工作。完成这项工作后, 你的应用必须调用 endBackgroundTask: 方法才能让系统进入 知道它已经完成并且可以暂停。
因此,您需要在此处执行有限长度的任务,同时挂起应用。这将为您的应用购买足够的时间来将您的记录同步到服务器。
示例代码段:
import UIKit
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
var backgroundTask: UIBackgroundTaskIdentifier!
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
// Override point for customization after application launch.
return true
}
func applicationWillResignActive(_ application: UIApplication) {
self.registerBackgroundTask()
// Do your background work here
print("Do your background work here")
// end the task when work is completed
self.endBackgroundTask()
}
func registerBackgroundTask() {
self.backgroundTask = UIApplication.shared.beginBackgroundTask { [weak self] in
self?.endBackgroundTask()
}
assert(self.backgroundTask != UIBackgroundTaskInvalid)
}
func endBackgroundTask() {
print("Background task ended.")
UIApplication.shared.endBackgroundTask(backgroundTask)
backgroundTask = UIBackgroundTaskInvalid
}
}
有关详细信息,请参阅此文章。