json - 编码/解码时 UIImage 不等效

标签 json swift uiimage jsondecoder equatable

我一直在对我的模型进行一些测试,以确保当我将它们编码为 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
}
首先,这是我如何制作 UIImageCodable 一起工作:
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)
    }
}
此外,ProfileImageEquatable一致性使用 isEqual(_:) UIImage属性,其中they say是“确定两幅图像是否包含相同图像数据的唯一可靠方法”。
然而,我的测试仍然失败,我不知道为什么。任何帮助将不胜感激。

最佳答案

the only reliable way to determine whether two images contain the same image data


他们错了。那篇文档过去也误导了我!
比较两个图像内容(底层位图)是否相等的方法是比较它们的 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

关于json - 编码/解码时 UIImage 不等效,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68137401/

相关文章:

ios - SKTexture 和 UIImage 的缩放属性

android - 如何编辑(更新)JSON 文件中的数据 flutter

javascript - 在 JavaScript 中更改时下拉列表值为空

ios - 从嵌套在 TableViewCell 中的 CollectionViewCell 中呈现 ViewController

swift - 我想在解析时保存 UIImage 而不是 nil 值

ios - UIImage 中的绘图被扭曲

objective-c - JSONModel 返回 nil

php - 来自PHP服务的JSON解析问题

ios - 允许编辑 UIImagePickerViewController 无法选择 2 个不同的图像

swift - 如何将 Firestore 文档映射到结构?