问题描述
我正在修改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)
}
}
}