本节书摘来自异步社区《Swift开发实战》一书中的第2章,第2.6节枚举和结构体,作者 李宁,更多章节内容可以访问云栖社区“异步社区”公众号查看
2.6 枚举和结构体
在Swift语言中,使用关键字enum来创建一个枚举。就像类和其他所有命名类型一样,在枚举中可以包含方法。例如,如下所示的演示代码。
enum Rank: Int {
case Ace = 1
case Two, Three, Four, Five, Six, Seven, Eight, Nine, Ten
case Jack, Queen, King
func simpleDescription() -> String {
switch self {
case .Ace:
return "ace"
case .Jack:
return "jack"
case .Queen:
return "queen"
case .King:
return "king"
default:
return String(self.toRaw())
}
}
}
let ace = Rank.Ace
let aceRawValue = ace.toRaw()
在上面的演示代码中,因为枚举原始值的类型是Int,所以只需要设置第一个原始值即可,剩下的原始值会按照顺序赋值。另外,也可以使用字符串或者浮点数作为枚举的原始值。
接下来可以使用函数toRaw和函数fromRaw在原始值和枚举值之间进行转换,例如,如下所示的演示代码。
if let convertedRank = Rank.fromRaw(3) {
let threeDescription = convertedRank.simpleDescription()
}
在Swift语言中,枚举的成员值是实际值,并不是原始值的另一种表达方法。实际上,如果原始值没有意义,则不需要进行任何设置。例如,如下所示的演示代码。
enum Suit {
case Spades, Hearts, Diamonds, Clubs
func simpleDescription() -> String {
switch self {
case .Spades:
return "spades"
case .Hearts:
return "hearts"
case .Diamonds:
return "diamonds"
case .Clubs:
return "clubs"
}
}
}
let hearts = Suit.Hearts
let heartsDescription = hearts.simpleDescription()
在Swift语言中,有两种方式可以引用Hearts成员:给hearts常量赋值时,枚举成员Suit.Hearts需要用全名来引用,因为常量没有显式指定类型。在switch里,枚举成员使用缩写.Hearts来引用,因为self的值已经知道是一个suit。已知变量类型的情况下可以使用缩写。
在Swift语言中,使用关键字struct来创建一个结构体。结构体和类有很多相同的地方,比如方法和构造器。结构体和类之间最大的一个区别就是结构体是传值,类是传引用。例如,如下所示的演示代码。
struct Card {
var rank: Rank
var suit: Suit
func simpleDescription() -> String {
return "The \(rank.simpleDescription()) of \
(suit.simpleDescription())"
}
}
let threeOfSpades = Card(rank: .Three, suit: .Spades)
let threeOfSpadesDescription = threeOfSpades.simpleDescription()
一个枚举成员的实例可以有实例值,相同枚举成员的实例可以有不同的值,只需在创建实例时传入值即可。实例值和原始值是不同的,枚举成员的原始值对于所有实例都是相同的,而且是在定义枚举的时候设置原始值。
例如,可以考虑从服务器获取日出和日落的时间,这样服务器会返回正常结果或者错误信息。例如,如下所示的演示代码。
enum ServerResponse {
case Result(String, String)
case Error(String)
}
let success = ServerResponse.Result("6:00 am", "8:09 pm")
let failure = ServerResponse.Error("Out of cheese.")
switch success {
case let .Result(sunrise, sunset):
let serverResponse = "Sunrise is at \(sunrise) and sunset is at \(sunset)."
case let .Error(error):
let serverResponse = "Failure... \(error)"
}