【Swift】同时使用RawRepresentable和Codable的注意点

AppStorage相信很多人用过,相当于UserDefaults的属性包装器,但比UserDefaults更好用的是能在SwiftUI中作为一个响应式变量同步刷新视图。

起因

想使用AppStorage来存储一些比较简单的自定义模型,例如:

swift 复制代码
struct Cat {
    var name: String
    var age: Int
}

首先,想存储自定义模型到AppStorage中,就得遵守RawRepresentable,并且RawValue要为StringInt类型。模型转String一般就是转成JSON 字符串,因此很自然会想到Codable,使用JSONEncoder编码和JSONDecoder解码:

swift 复制代码
extension Cat: Codable, RawRepresentable {
    var rawValue: String {
        let jsonData = try! JSONEncoder().encode(self)
        return String(data: jsonData, encoding: .utf8)!
    }
    
    init?(rawValue: String) {
        let data = rawValue.data(using: .utf8)!
        self = try! JSONDecoder().decode(Self.self, from: data)
    }
}

补充说明,遵守Codable的模型,如果所有属性都是已经遵守了Codable,那么就不用自己去实现endodedecode

  • PS: Swift提供的基本类型都是Codable的,如StringInt,还有ArrayDictionary(只要存储的元素也是Codable的即可)。

写完,编译没问题,运行,卡死!!!

What the hell?

直接揭晓答案,这是因为死循环 了,看看swift的源码:

可以看出,只要遵守了CodableRawRepresentable,其默认实现的endode里面会调用rawValue,而rawValue是刚刚自己重写 的计算属性,内部则调用了JSONEncoder().encode(self),因此造成死循环了。

解决方法

自己去实现endodedecode

swift 复制代码
extension Cat: Codable {
    enum CodingKeys: String, CodingKey {
        case name, age
    }
    
    init(from decoder: Decoder) throws {
        let c = try decoder.container(keyedBy: CodingKeys.self)
        name = try c.decode(String.self, forKey: .name)
        age = try c.decode(Int.self, forKey: .age)
    }
    
    func encode(to encoder: Encoder) throws {
        var c = encoder.container(keyedBy: CodingKeys.self)
        try c.encode(name, forKey: .name)
        try c.encode(age, forKey: .age)
    }
}

打破循环!

最后

这个问题最主要的原因是过于依赖Swift的便捷性(默认实现Codable),真没想到CodableRawRepresentable一同使用时,会被这些默认实现导致死循环,还好这种问题开发时就能发现,以后使用新东西还是要多做测试,防止这种问题上线后才发现。

总结一下:使用AppStorage存储自定义模型,模型需要遵守RawRepresentable并且RawValue要为String,如果使用Codable进行模型转换,要自己去实现endodedecode防止死循环。

相关推荐
zhyongrui15 小时前
SnipTrip 发热优化实战:从 60Hz 到 30Hz 的性能之旅
ios·swiftui·swift
大熊猫侯佩1 天前
Neo-Cupertino 档案:撕开 Actor 的伪装,回归 Non-Sendable 的暴力美学
swift·observable·actor·concurrency·sendable·nonsendable·data race
大熊猫侯佩1 天前
赛博深渊(上):用 Apple Foundation Models 提炼“禁忌知识”的求生指南
llm·swiftui·大语言模型·foundationmodel·apple ai·apple 人工智能·summarize
zhyongrui3 天前
SwiftUI 光晕动画性能优化:消除托盘缩放卡顿的实战方案
ios·性能优化·swiftui
2501_915921433 天前
在没有源码的前提下,怎么对 Swift 做混淆,IPA 混淆
android·开发语言·ios·小程序·uni-app·iphone·swift
00后程序员张3 天前
对比 Ipa Guard 与 Swift Shield 在 iOS 应用安全处理中的使用差异
android·开发语言·ios·小程序·uni-app·iphone·swift
大熊猫侯佩4 天前
星际穿越:SwiftUI 如何让 ForEach 遍历异构数据(Heterogeneous)集合
swiftui·swift·遍历·foreach·any·异构集合·heterogeneous
符哥20084 天前
对比ArkTsUI和Flutter和 SwiftUI 和Jetpack Compose四个框架语法及使用场景。
flutter·ios·swiftui
hjs_deeplearning4 天前
认知篇#15:ms-swift微调中gradient_accumulation_steps和warmup_ratio等参数的意义与设置
开发语言·人工智能·机器学习·swift·vlm
墨瑾轩4 天前
C# PictureBox:5个技巧,从“普通控件“到“图像大师“的蜕变!
开发语言·c#·swift