枚举与 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 组合类型指定函数参数的类型?