首页 > 解决方案 > 可编码一致性结构的 @Published 数组的可编码一致性

问题描述

我正在修改 URLSession,本质上是在构建一个超级简单的应用程序来将 JSON 文件加载到 ContentView 中,有点像 Facebook 的朋友列表,并且想要弄清楚我遇到的任何错误,而是 Swift 的内部工作Codable协议。下面是一些代码和解释:

struct User: Identifiable, Codable {
    struct Friend : Identifiable, Codable {
        var name : String
        var id : String
    }

    var id : String
    var isActive: Bool
    var name : String
    var age: Int
    var company : String
    var address : String
    var about : String
    var registered : String
    
    var friends : [Friend]
    
    var checkIsActive: String {
        return self.isActive ? "" :""
    }
    
}

综上所述,我有一个User结构,其中包含一堆符合Codable.

class UsersArrayClass: ObservableObject {
    @Published var userArray = [User]() 
}

但是,我有另一个类,UsersArrayClass它创建一个@Published var userArray结构User对象。此类符合@ObservableObject协议,但是当然,当我尝试使其符合 时Codable,它可能不喜欢那样,因为@Published属性包装器被应用于数组本身......这实际上是让我感到困惑的地方,但如果用户结构具有Codable一致性,为什么userArray包含User对象的哪个不自动符合Codable

我在想也许将所有这些加载到核心数据模型中可以解决我的问题,但除非我明白我在这里错过了什么,否则我仍然无法继续前进,所以提前感谢任何输入。

标签: swiftstructswiftuicodableurlsession

解决方案


这很 hacky,但我们可以通过扩展添加Codable一致性,Published尽管缺乏对Published内部的访问。

extension Published: Codable where Value: Codable {
    public func encode(to encoder: Encoder) throws {
        guard
            let storageValue =
                Mirror(reflecting: self).descendant("storage")
                .map(Mirror.init)?.children.first?.value,
            let value =
                storageValue as? Value
                ??
                (storageValue as? Publisher).map(Mirror.init)?
                .descendant("subject", "currentValue")
                as? Value
        else { fatalError("Failed to encode") }
        
        try value.encode(to: encoder)
    }
    
    public init(from decoder: Decoder) throws {
        self.init(initialValue: try .init(from: decoder))
    }
}

快速检查:

class User: ObservableObject, Codable {
    @Published var name = "Paul"
}

struct ContentView: View {
    @ObservedObject var user = User()
    var body: some View {
        let data = try? JSONEncoder().encode(user)
        let dataFromStr = """
                {
                    "name": "Smith"
                }
                """
            .data(using: .utf8)
        let decoded = try! JSONDecoder().decode(User.self, from: dataFromStr!)
        return
            VStack{
                Text(verbatim: String(data: data!, encoding: .utf8) ?? "encoding failed")
                Text(decoded.name)
            }
    }
}

推荐阅读