UIImage 在编码/解码时不等效

Pra*_*ech 6 json uiimage swift equatable jsondecoder

我一直在对我的模型进行一些测试,以确保当我将它们编码为 JSON 然后使用JSONEncoder/Decoder. 然而,我的一项测试失败了,罪魁祸首是UIImage。我已确保在编码/解码过程中没有抛出任何错误。

首先,这是有问题的测试:

func testProfileImageCodable() throws {
    let image = ProfileImage(UIImage(systemName: "applelogo")!)
    try XCTAssertTrue(assertCodable(image))
}
Run Code Online (Sandbox Code Playgroud)

这是我的“可编码性”测试,我确保编码/解码之前和之后类型相等:

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
}
Run Code Online (Sandbox Code Playgroud)

首先,这是我的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")
            )
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

生活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)
    }
}
Run Code Online (Sandbox Code Playgroud)

此外,ProfileImageEquatable一致性使用了isEqual(_:)UIImage属性,他们称这是“确定两个图像是否包含相同图像数据的唯一可靠方法”。

然而,我的测试仍然失败,我不知道为什么。任何帮助将不胜感激。

mat*_*att 7

确定两个图像是否包含相同图像数据的唯一可靠方法

他们的想法是错误的。那篇文档过去也误导了我!

比较两个图像内容(底层位图)是否相等的方法是比较它们的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)
    }
}
Run Code Online (Sandbox Code Playgroud)

这是我的测试:

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
Run Code Online (Sandbox Code Playgroud)