swift – 你如何反映Codable / CodableKeys协议的设计?
发布时间:2020-12-14 04:32:15 所属栏目:百科 来源:网络整理
导读:我试图实现类似于 Swift如何在实现Codable的类中定义的枚举上使用CodableKeys协议集的方法.在我的例子中,该类是CommandHandler,枚举是CommandIds,它不需要编译器的代码生成,因为枚举将始终显式指定. 这是我追求的简化版本…… protocol CommandId{}protocol
我试图实现类似于
Swift如何在实现Codable的类中定义的枚举上使用CodableKeys协议集的方法.在我的例子中,该类是CommandHandler,枚举是CommandIds,它不需要编译器的代码生成,因为枚举将始终显式指定.
这是我追求的简化版本…… protocol CommandId{} protocol CommandHandler{ associatedtype CommandIds : CommandId,RawRepresentable } class HandlerA : CommandHandler{ enum CommandIds : String,CommandId{ case commandA1 case commandA2 } } class HandlerB : CommandHandler{ enum CommandIds : String,CommandId{ case commandB1 case commandB2 case commandB3 } } func processHandler<T:CommandHandler>(_ handler:T){ // Logic to iterate over CommandIds. <-- This is where I get stumped } let handlerA = HandlerA() processHandler(handlerA) 我正在努力处理processHandler中的代码,因为我不确定如何从处理程序实例中获取枚举值. 那我错过了什么?获取相关枚举值的代码是什么? 解决方法
好的,我相信我已经准备好了所有的部分来展示你如何在Swift中做到这一点.事实证明,我修改过的问题正好处于如何做到这一点的正确边缘.
这是我在Swift 4中编写的示例… 首先,这是您如何定义使其工作所需的协议.从设计的角度来看,它们分别与CodableKeys和Codable同义. protocol CommandId : EnumerableEnum,RawRepresentable {} protocol CommandHandler{ associatedtype CommandIds : CommandId } 这是一个协议及其相关的扩展,使枚举的“case”值可枚举.您只需使您的枚举符合EnumerableEnum协议,即可获得“值”数组. 由于上面的CommandId协议已经应用于有问题的枚举,我们通过使EnumerableEnum协议在其自己的定义中应用来简化.这样我们只需要将CommandId应用到我们的枚举中,我们就可以得到它们. public protocol EnumerableEnum : Hashable { static var values: [Self] { get } } public extension EnumerableEnum { public static var values: [Self] { let valuesSequence = AnySequence { () -> AnyIterator<Self> in var caseIndex = 0 return AnyIterator { let currentCase: Self = withUnsafePointer(to: &caseIndex){ $0.withMemoryRebound(to: self,capacity: 1){ $0.pointee } } guard currentCase.hashValue == caseIndex else { return nil } caseIndex += 1 return currentCase } } return Array(valuesSequence) } } 这是实现我的CommandHandler / CommandId协议的两个类 class HandlerA : CommandHandler{ enum CommandIds : Int,CommandId{ case commandB1 = "Command B1" case commandB2 case commandB3 = "Yet another command" } } 这是一个接受CommandHandler类型的测试函数 func enumerateCommandIds<T:CommandHandler>(_ commandHandlerType:T.Type){ for value in commandHandlerType.CommandIds.values{ let caseName = String(describing:value) let caseRawValue = value.rawValue print("(caseName) = '(caseRawValue)'") } } 最后,这是运行该测试的结果 enumerateCommandIds(HandlerA.self) // Outputs // commandA1 = '0' // commandA2 = '1' enumerateCommandIds(HandlerB.self) // Outputs // commandB1 = 'Command B1' // commandB2 = 'commandB2' // commandB3 = 'Yet another command' 到达这里是漫长而多风的道路,但我们做到了!感谢大家的帮助! (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |