微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

@published 不适用于类型数组,但适用于单一类型

如何解决@published 不适用于类型数组,但适用于单一类型

我正在测试非常简单的代码,有一个类确认 @ObservableObject 协议,并且有两个 @Published 属性一个是我的自定义 type 对象数组,另一个是只是同班的普通 type我有两种观点,其中 parent 拥有单一的事实来源,它有一个 List一个 navigationLink。 NavigationLink 将 @Published 数组对象作为 Binding 传递给 subView。

问题-:

当我输入 TextField 时,视图没有更新。它正在修改 @Published 对象属性,但不更新子视图。这仅在我使用 Array 个对象进行测试时发生,但现在如果我注释 array 个对象属性并简单地使用另一个(没有数组),它会更新得很好。

谁能解释一下背后的原因,这是一个错误吗?还是 Array 在幕后做其他事情?

不工作的例子-:

import SwiftUI


struct Person: Identifiable {
    let id: String = UUID().uuidString
    var name: String = "Tushar"
    var age: Int = 25
    
    init(name: String,age: Int) {
        self.name = name
        self.age = age
    }
}

 class PersonStore: ObservableObject {
    @Published var name: [Person] = [Person(name:"foo",age:23),Person(name:"foobar",age:23)]
    //@Published var name: Person = Person(name:"foo",age:23)
}

struct ContentViewsss: View {
    
    @Observedobject var store = PersonStore()
    
    @State var isEditing = false
    
    var body: some View {
        NavigationView{
            List(store.name.indices){ index in
                NavigationLink(destination: ContentViewsss1(store: $store.name[index])) {
                    Text("Edit")
                }
            }.navigationTitle(“Welcome")
        }
    }
}

struct ContentViewsss1: View {
    
    @Binding var store: Person

    var body: some View {
        TextField("My Text",text: $store.name)
        Text(store.name)
    }
}

工作示例-:

import SwiftUI


struct Person: Identifiable {
    let id: String = UUID().uuidString
    var name: String = "Tushar"
    var age: Int = 25
    
    init(name: String,age: Int) {
        self.name = name
        self.age = age
    }
}

 class PersonStore: ObservableObject {
   // @Published var name: [Person] = [Person(name:"foo",age:23)]
    @Published var name: Person = Person(name:"foo",age:23)
}

struct ContentViewsss: View {
    
    @Observedobject var store = PersonStore()
    
    @State var isEditing = false
    
    var body: some View {
        NavigationView{
            List{ 
                NavigationLink(destination: ContentViewsss1(store: $store.name)) {
                    Text("Edit")
                }
            }.navigationTitle("Welcome")
        }
    }
}

struct ContentViewsss1: View {
    
    @Binding var store: Person

    var body: some View {
        TextField("My Text",text: $store.name)
        Text(store.name)
    }
}

解决方法

据说,导致问题的不是数组——而是List。您会在原始代码中看到,如果将 List 更改为 ForEach,它将起作用。我不完全确定故障发生在哪里,而且它比人们想象的更微妙。

我的建议是使用中间视图模型。以下代码有效:

class ViewModel: ObservableObject {
    @Published var person : Person {
        didSet {
            binding.wrappedValue = person
        }
    }
    var binding : Binding<Person>
    
    init(withPersonBinding person: Binding<Person>) {
        self.person = person.wrappedValue
        self.binding = person
    }
}

struct ContentViewsss1: View {
    //@Binding var store: Person
    @ObservedObject var store : ViewModel
    
    var body: some View {
        TextField("My Text",text: $store.person.name)
        Text(store.person.name)
    }
}

被这样调用:

NavigationLink(destination: ContentViewsss1(store: ViewModel(withPersonBinding: $store.name[index]))) {
   Text("Edit")
}

现在,这是非常奇怪的一点。如果您返回原始代码并将详细信息视图修改为:

struct ContentViewsss1: View {
    
    @Binding var store: Person
    @State private var txt = ""
    
    var body: some View {
        TextField("My Text",text: $store.name)
        Text(store.name)
        TextField("Other text",text: $txt)
    }
}

您会看到,如果您编辑第一个文本字段,您的更改将不会得到反映(正如您所经历的那样)。 但是,只要您更改了第二个字段中的文本,@State 中的更改就会触发它也检查@Binding,并且该字段会更新。同样,这仅发生在 List 而不是 ForEach。 SwiftUI 的奥秘...

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。