我有几个可编码的结构,我想创建一个通用协议,将它们编码为CKRecord for CloudKit并进行解码.
extension encodable { var dictionary: [String: Any] { return (try? JSONSerialization.jsonObject(with: JSONEncoder().encode(self),options: .allowFragments)) as? [String: Any] ?? [:] } }
然后在协议扩展中,我将记录创建为属性,如果类型为Data,我尝试创建CKAsset.
var ckEncoded: CKRecord? { // Convert self.id to CKRecord.name (CKRecordID) guard let idString = self.id?.uuidString else { return nil } let record = CKRecord(recordtype: Self.entityType.rawValue,recordID: CKRecordID(recordName: idString)) self.dictionary.forEach { if let data = $0.value as? Data { if let asset: CKAsset = try? ckAsset(from: data,id: idString) { record[$0.key] = asset } } else { record[$0.key] = $0.value as? CKRecordValue } } return record }
要解码:
func decode(_ ckRecord: CKRecord) throws { let keyIntersection = Set(self.dtoEncoded.dictionary.keys).intersection(ckRecord.allKeys()) var dictionary: [String: Any?] = [:] keyIntersection.forEach { if let asset = ckRecord[$0] as? CKAsset { dictionary[$0] = try? self.data(from: asset) } else { dictionary[$0] = ckRecord[$0] } } guard let data = try? JSONSerialization.data(withJSONObject: dictionary) else { throw Errors.LocalData.isCorrupted } guard let dto = try? JSONDecoder().decode(self.DTO,from: data) else { throw Errors.LocalData.isCorrupted } do { try decode(dto) } catch { throw error } }
除数据类型外,所有内容都可以正常工作.它无法从字典中识别出来.所以,我无法将其转换为CKAsset.先感谢您.
解决方法
我还发现到目前为止苹果还没有对此提供干净的支持.
我的解决方案是手动编码/解码:在我的Codable子类上,我添加了两个方法:
/// Returns CKRecord func ckRecord() -> CKRecord { let record = CKRecord(recordtype: "MyClasstype") record["title"] = title as CKRecordValue record["color"] = color as CKRecordValue return record } init(withRecord record: CKRecord) { title = record["title"] as? String ?? "" color = record["color"] as? String ?? kDefaultColor }
针对更复杂案例的另一个解决方案是使用一些第三方库,我遇到的是:https://github.com/insidegui/CloudKitCodable
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。