符合可编码结构的 @Published 数组的可编码一致性
Codable conformance at @Published array of a codable-conformed Struct
我正在研究 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]()
}
但是,我还有另一个 class、UsersArrayClass
,它创建 @Published var userArray
个 User
结构对象。这个 class 符合 @ObservableObject
协议,但是当然,当我试图让它符合 Codable
时,它可能不喜欢那样,因为 @Published
属性 包装器被应用于数组本身...这基本上让我感到困惑,尽管如果 User 结构具有 Codable
一致性,为什么 userArray
不包含 User
对象也自动符合 Codable
?
我在想也许将所有这些加载到核心数据模型中可以解决我的问题,但我仍然无法继续前进,除非我明白我在这里遗漏了什么,所以提前感谢任何输入。
/*
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 {}
它是 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)
}
}
}
我正在研究 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]()
}
但是,我还有另一个 class、UsersArrayClass
,它创建 @Published var userArray
个 User
结构对象。这个 class 符合 @ObservableObject
协议,但是当然,当我试图让它符合 Codable
时,它可能不喜欢那样,因为 @Published
属性 包装器被应用于数组本身...这基本上让我感到困惑,尽管如果 User 结构具有 Codable
一致性,为什么 userArray
不包含 User
对象也自动符合 Codable
?
我在想也许将所有这些加载到核心数据模型中可以解决我的问题,但我仍然无法继续前进,除非我明白我在这里遗漏了什么,所以提前感谢任何输入。
/*
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 {}
它是 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)
}
}
}