首页 > 解决方案 > 后台任务完成后 NavigationLink deinit ObservedObject

问题描述

SecondVM每次我从我的新视图推送ContentViewContentVM完成他的工作时,我都会遇到解除分配的问题。

描述

推送到SecondView 后,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)
        }
      }   
   }
}

并且SecondSecondVM

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。如何避免这种行为?

在此处输入图像描述

标签: swiftuicombineswiftui-navigationlink

解决方案


这是解决方案 - 使链接目标相等,因此当 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

推荐阅读