有时在 Swift 中,为委托JSONDecoder或工厂方法的类编写初始化程序可能会很方便。例如,一个人可能想写
final class Test: Codable {
let foo: Int
init(foo: Int) {
self.foo = foo
}
func jsonData() throws -> Data {
try JSONEncoder().encode(self)
}
convenience init(fromJSON data: Data) throws {
self = try JSONDecoder().decode(Self.self, from: data)
}
}
let test = Test(foo: 42)
let data = try test.jsonData()
let decodedTest = try Test(fromJSON: data)
print(decodedTest.foo)
Run Code Online (Sandbox Code Playgroud)
但这无法编译
Cannot assign to value: 'self' is immutable.
Run Code Online (Sandbox Code Playgroud)
解决此问题的解决方案是什么?
首先,请注意此限制仅适用于classes,因此示例初始化程序将按原样适用于structs 和enums,但并非所有情况都允许将 a 更改class为这些类型之一。
这种对class初始化器的限制是一个经常出现在这个站点上的痛点(例如)。在 Swift 论坛上有一个线程讨论这个问题,并且已经开始添加必要的语言特性来编译上面的示例代码,但这在 Swift 5.4 中还没有完成。从线程:
Swift 自己的标准库和 Foundation 覆盖层通过使类符合虚拟协议并在必要时使用协议扩展初始化器来实现此功能来绕过此缺失的功能。
使用这个想法来修复示例代码会产生
final class Test: Codable {
let foo: Int
init(foo: Int) {
self.foo = foo
}
func jsonData() throws -> Data {
try JSONEncoder().encode(self)
}
}
protocol TestProtocol: Decodable {}
extension Test: TestProtocol {}
extension TestProtocol {
init(fromJSON data: Data) throws {
self = try JSONDecoder().decode(Self.self, from: data)
}
}
let test = Test(foo: 42)
let data = try test.jsonData()
let decodedTest = try Test(fromJSON: data)
print(decodedTest.foo)
Run Code Online (Sandbox Code Playgroud)
这工作正常。如果Test是唯一符合 的类型TestProtocol,那么只会Test得到这个初始化器。
另一种方法是简单地扩展Decodable或您的类符合的其他协议,但如果您不希望符合该协议的其他类型也获得您的初始化程序,这可能是不可取的。