如何反映可编码/可编码密钥协议的设计?

2024-02-29

我正在尝试实现类似于 Swift 如何利用CodableKeys在实现的类中定义的枚举上设置的协议Codable。就我而言,班级是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 commandA1
        case commandA2
    }
}

class HandlerB : CommandHandler{
    enum CommandIds : String, CommandId{
        case commandB1 = "Command B1"
        case commandB2
        case commandB3 = "Yet another command"
    }
}

这是一个测试函数,它接受CommandHandler type

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'

到达这里的路很长,风很大,但我们做到了!感谢大家的帮助!

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

如何反映可编码/可编码密钥协议的设计? 的相关文章

随机推荐