@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属性包装器已应用于数组本身...这实际上使我感到困惑,但是如果User结构具有Codable一致性,为什么包含userArray对象的User自动符合Codable还有吗?

我当时想也许可以将所有这些加载到Core Data模型中来解决我的问题,但是除非我了解我在这里遗漏的内容,否则我仍然无法继续前进,因此在此先感谢您的任何输入

解决方法

/*
 Cannot automatically synthesize 'Encodable' because 'Published<[User]>'
 does not conform to 'Encodable' @Published var userArray = [User]()
 */

// Published declaration
@propertyWrapper struct Published<Value> { ... }

已发布的内容当前不符合Codable或Foundation中的任何常见协议

尝试使Published符合Codeable导致的以下错误:

/*
 Implementation of 'Decodable' cannot be
 automatically synthesized in an extension in a different file to the type
 */
extension Published: Codable where Value: Codable {}
,

这很骇人,但尽管无法访问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)
            }
    }
}