swiftui - 后台任务完成后 NavigationLink deinit ObservedObject
问题描述
SecondVM
每次我从我的新视图推送ContentView
并ContentVM
完成他的工作时,我都会遇到解除分配的问题。
描述
推送到Second
View 后,ObservableObject
在任务ContentVM
完成后解除分配。
我的示例代码如下ContentView
和`ContentVM:
final class ContentVM: ObservableObject {
@Published var title = "Start"
init() {
DispatchQueue.main.asyncAfter(deadline: .now() + 5) {
self.title = "Changed"
}
}
}
struct ContentView: View {
@ObservedObject var vm = ContentVM()
var body: some View {
NavigationView {
VStack {
NavigationLink(destination: Second()) {
Text("Go To second")
}
Spacer()
.frame(height: 40)
Text(vm.title)
}
}
}
}
并且Second
和SecondVM
final class SecondVM: ObservableObject {
@Published var name: String = ""
func getName() {
DispatchQueue.main.asyncAfter(deadline: .now() + 2) {
self.name = "TEST"
}
}
}
struct Second: View {
@ObservedObject var vm = SecondVM()
var body: some View {
Text(vm.name)
.padding(50)
.background(vm.name.isEmpty ? Color.white : Color.black)
.foregroundColor(Color.white)
.onAppear {
self.vm.getName()
}
}
}
正如您在视频中看到的,只有当我按下Second
查看时才会出现问题。黑色矩形正确显示,然后在完成任务ContentVM
后,此黑色矩形因取消分配而消失SecondVM
。如何避免这种行为?
解决方案
这是解决方案 - 使链接目标相等,因此当 ContentView 根据自己的状态更改更新时,它不会重新创建目标视图(否则会发生这种情况并且是观察到问题的根源)。
使用 Xcode 11.5b2 测试
// in ContentView, id can be any type but constant in this case
NavigationLink(destination: Second(id: 1).equatable()) {
Text("Go To second")
}
// SecondView
struct Second: View, Equatable {
let id: Int
static func == (lhs: Second, rhs: Second) -> Bool {
lhs.id == rhs.id
}
@ObservedObject var vm = SecondVM()
// .. other code