枚举与 RawRepresentable 的一致性

Posted

技术标签:

【中文标题】枚举与 RawRepresentable 的一致性【英文标题】:enum conformance to RawRepresentable 【发布时间】:2019-04-03 07:44:32 【问题描述】:

在 Swift 书中,枚举示例运行良好

enum CompassPoint: String 
    case north, south, east, west


var northCom = CompassPoint.north
print (northCom)

但是我想使用一个可失败的初始化器,所以做了一个例子

enum WeekDay: String 
    case monday, tuesday, wednesday, thursday, friday
    init?(rawValue: Int)
        switch rawValue 
        case 0 : self = .monday
        case 1 : self = .tuesday
        case 2 : self = .wednesday
        case 3 : self = .thursday
        case 4 : self = .friday
        default : return nil
        
    

并得到一个错误,即 Weekday 不符合 RawRepresentable - 虽然我觉得应该由编译器综合,所以不明白为什么这不能编译。

我做了什么 创建了一个类似的工作示例(以查看问题所在),但我仍然想使用带有可失败初始化程序的枚举来遵守 RawRepresentable。我在 Swift 书籍、Stack Overflow 问题或更广泛的互联网上找不到这样的例子。

我提供的内容给出了上面的完整示例,其中包含错误以及我期望的行为的工作示例。

什么没有帮助通过链接或评论引用 Swift 书籍没有帮助,因为我从那里举了一个例子。我想在那个例子的基础上使用一个可失败的初始化程序。不使用枚举或可失败的初始化程序也无济于事。问题是关于使用带有枚举的可失败初始化程序,并符合 RawRepresentable。这不是家庭作业,但这些是我自己学习的问题的限制,我对结果很感兴趣。

问题如何使用一个可失败的初始化程序,以及非工作示例中的枚举?

【问题讨论】:

【参考方案1】:

显然您对init?(rawValue: Int) 的定义阻止了编译器自动推断RawValue 类型。添加类型别名有帮助:

enum WeekDay: String 
    typealias RawValue = String

    case monday, tuesday, wednesday, thursday, friday

    init?(rawValue: Int)
        switch rawValue 
        case 0 : self = .monday
        case 1 : self = .tuesday
        case 2 : self = .wednesday
        case 3 : self = .thursday
        case 4 : self = .friday
        default : return nil
        
    

或者使用不同的参数名称定义您的自定义初始化函数:

enum WeekDay: String 

    case monday, tuesday, wednesday, thursday, friday

    init?(rawInt: Int)
        switch rawInt 
        case 0 : self = .monday
        case 1 : self = .tuesday
        case 2 : self = .wednesday
        case 3 : self = .thursday
        case 4 : self = .friday
        default : return nil
        
    

【讨论】:

我现在可以看到,可以称之为索引 - 从使用你的面包中看到 - useyourloaf.com/blog/enum-raw-values-and-failable-initializers【参考方案2】:

“虽然我觉得一致性应该由编译器综合”

是的,您的rawValue 是字符串类型,而不是 Int。只需像这样创建您的枚举:

enum WeekDay: String 
    case monday, tuesday, wednesday, thursday, friday

然后像这样创建一个WeekDay

let monday: WeekDay? = WeekDay(rawValue: "monday")
let notADay: WeekDay? = WeekDay(rawValue: "foo")

当然你也可以添加一个自定义的init,它接受一个整数作为参数:

enum WeekDay: String 
    case monday, tuesday, wednesday, thursday, friday
    init?(integer: Int)
        switch integer 
        case 0 : self = .monday
        case 1 : self = .tuesday
        case 2 : self = .wednesday
        case 3 : self = .thursday
        case 4 : self = .friday
        default : return nil
        
    

然后像这样创建你的工作日:

let monday: WeekDay? = WeekDay(integer: 0)
let notADay: WeekDay? = WeekDay(integer: 30)

【讨论】:

【参考方案3】:

从 Swift 5 开始,除了合成的 init?(rawValue: String) 之外,还有第二个 init?(rawValue: Int) 没有错误。但是,如果您想覆盖 RawRepresantable 实现并让 rawValue 成为 Int,这是可能的。

extension WeekDay : RawRepresentable 
    
    typealias RawValue = Int
    init?(rawValue: Int)
        switch rawValue 
        case 0 : self = .monday
        case 1 : self = .tuesday
        case 2 : self = .wednesday
        case 3 : self = .thursday
        case 4 : self = .friday
        default : return nil
        
    
    
    var rawValue: Int 
        switch self 
        case .monday : return 0
        case .tuesday :  return 1
        case .wednesday : return 2
        case .thursday : return 3
        case .friday : return 4
        
    

print(WeekDay.init(rawValue: 2)) // Optional(WeekDay.wednesday)
print(WeekDay.RawValue.self) // Int

【讨论】:

以上是关于枚举与 RawRepresentable 的一致性的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 CustomStringConvertible 和 RawRepresentable 组合类型指定函数参数的类型?

如何使用通用默认参数

swift学习笔记五.使用枚举优雅的管理Segue

与枚举类型的属性的可解码一致性

枚举GC Roots的实现

Java枚举类型在RPC调用时由于版本不一致遇见的问题