问题描述
我一直在对我的模型进行一些测试,以确保当我将它们编码为 JSON 然后使用 JSONEncoder/Decoder
将它们解码回来时它们是相等的。但是,我的一项测试失败了,罪魁祸首是 UIImage
。我确保在编码/解码过程中没有抛出任何错误。
首先,这是有问题的测试:
func testProfileImageCodable() throws {
let image = ProfileImage(UIImage(systemName: "applelogo")!)
try XCTAssertTrue(assertCodable(image))
}
这是我的“可编码性”测试,我确保编码/解码前后的类型相同:
func assertCodable<T: Codable & Equatable>(
_ value: T,decoder: JSONDecoder = .init(),encoder: JSONEncoder = .init()
) throws -> Bool {
let encoded = try encoder.encode(value)
let decoded = try decoder.decode(T.self,from: encoded)
return value == decoded
}
首先,以下是我如何让 UIImage
与 Codable
一起工作:
extension KeyedEncodingContainer {
mutating func encode(_ value: UIImage,forKey key: Key) throws {
guard let data = value.pngData() else {
throw EncodingError.invalidValue(
value,EncodingError.Context(codingPath: [key],debugDescription: "Failed convert UIImage to data")
)
}
try encode(data,forKey: key)
}
}
extension KeyedDecodingContainer {
func decode(_ type: UIImage.Type,forKey key: Key) throws -> UIImage {
let imageData = try decode(Data.self,forKey: key)
if let image = UIImage(data: imageData) {
return image
} else {
throw DecodingError.dataCorrupted(
DecodingError.Context(codingPath: [key],debugDescription: "Failed load UIImage from decoded data")
)
}
}
}
UIImage
存在于 ProfileImage
类型中,因此使其符合 Codable
如下所示:
extension ProfileImage: Codable {
enum CodingKeys: CodingKey {
case image
}
public init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
self.image = try container.decode(UIImage.self,forKey: .image)
}
public func encode(to encoder: Encoder) throws {
var container = encoder.container(keyedBy: CodingKeys.self)
try container.encode(self.image,forKey: .image)
}
}
此外,ProfileImage
的 Equatable
一致性在 UIImage
属性上使用 isEqual(_:)
,they say 是“确定两个图像是否相同的唯一可靠方法包含相同的图像数据。”
然而,我的测试仍然失败,我不知道为什么。任何帮助将不胜感激。
解决方法
判断两张图片是否包含相同图片数据的唯一可靠方法
他们错了。过去那篇文档也误导了我!
比较两个图像内容(底层位图)是否相等的方法是比较它们的 pngData
。
然而,在最深层次上,您的代码有什么问题,就是 UIImage 包含 scale
信息,而您却丢弃了这些信息。例如,您的原始图像的 scale
可能是 2 或 3。但是当您在解码时调用 image(data:)
时,您没有考虑到这一点。如果您确实考虑到了这一点,那么您的断言就会如您所愿。
我像这样调整了您的代码(可能有更好的方法,我只是想证明 scale
是问题所在):
struct Image: Codable {
let image:UIImage
init(image:UIImage) {
self.image = image
}
enum CodingKeys: CodingKey {
case image
case scale
}
public init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
let scale = try container.decode(CGFloat.self,forKey: .scale)
let image = try container.decode(UIImage.self,forKey: .image)
self.image = UIImage(data:image.pngData()!,scale:scale)!
}
public func encode(to encoder: Encoder) throws {
var container = encoder.container(keyedBy: CodingKeys.self)
try container.encode(self.image,forKey: .image)
try container.encode(self.image.scale,forKey: .scale)
}
}
这是我的测试:
let im = UIImage(systemName:"applelogo")!
let encoded = try! JSONEncoder().encode(Image(image:im))
let decoded = try! JSONDecoder().decode(Image.self,from: encoded)
assert(im.pngData()! == decoded.image.pngData()!)
print("ok") // yep