嵌套的类型
枚举经常被创建用来支撑特定类或者结构体的功能。类似的,为了在一个复合类型上下文中使用,定义纯粹的工具类和结构体也是可行的。为了做到这些,Swift允许定义嵌套类型,籍此可以在支持的类型定义中嵌套枚举、类、和结构体。
要把一个类型嵌套在另一个类型中,在那个类型的最外层大括号内定义类型就可以了。类型嵌套可以根据需要做任意层级的。
嵌套的类型实战
下面例子定义一个叫做BlackjackCard(译者注:扑克牌21点游戏)的结构体,它模拟了游戏21点使用的纸牌。BlackjackCard结构体包括两个嵌套枚举类型,分别叫做Suit和Rank。
在21点游戏中,纸牌A(译者注:纸牌A的全称是Ace,在21点游戏中是号牌,拿到此牌的玩家可以根据需要确定它的分值是1或者是11)的值或者是1或者是11。这个特性通过一个叫做Values的结构体实现,这个结构体嵌套在Rank枚举类型中:
struct BlackjackCard {
// nested Suit enumeration
enum Suit: Character {
case Spades = "♠", Hearts = "♡", Diamonds = "♢", Clubs = "♣"
}
// nested Rank enumeration
enum Rank: Int {
case Two = 2, Three, Four, Five, Six, Seven, Eight, Nine, Ten
case Jack, Queen, King, Ace
struct Values {
let first: Int, second: Int?
}
var values: Values {
switch self {
case .Ace:
return Values(first: 1, second: 11)
case .Jack, .Queen, .King:
return Values(first: 10, second: nil)
default:
return Values(first: self.rawValue, second: nil)
}
}
}
// BlackjackCard properties and methods
let rank: Rank, suit: Suit
var description: String {
var output = "suit is \(suit.rawValue),"
output += " value is \(rank.values.first)"
if let second = rank.values.second {
output += " or \(second)"
}
return output
}
}
Suit枚举描述了纸牌的四种花色,连同一个初始的Character值表示它们(花色)的符号。
Rank枚举描述了可能的13张牌的顺序,连同一个初始的Int值表示它们(顺序)的面值。(这个初始的Int值不适用于J(Jack)、Q(Queen)、K(King)和A(Ace))
像上面叙述的一样,Rank枚举定义了属于它自己的嵌套结构体叫做Values。这个结构体封装了这样一个现象:多数的纸牌只有一个值,但是A会有两个。Values结构定义了两个属性来表现这些:
first,是一个Int类型
second,是一个Int?(或者“可选Int”)
Rank还定义了一个计算属性,叫做values,它返回一个Values结构体的实例。这个计算属性根据纸牌的排序位置构造一个与排序位置对应的新的Values实例。对于Jack、Queen、King和Ace采用特殊值,对于数字牌,使用排序的初始Int值。
BlackjackCard结构体自身有两个属性——rank和suit。同时还有一个计算属性叫做description,这个计算属性使用rank和suit中存储的值创建一个针对这张牌的名字和值的描述信息。description属性使用可选绑定来检查是不是有第二个值需要显示,如果有,会将第二个值插入到描述信息中去。
因为BlackjackCard是一个没有自定义构造方法的结构体,所以它拥有一个隐式的成员构造方法,就像 结构体类型的成员构造方法(Memberwise Initializers for Structure Type) 一节描述的一样。可以用这个构造方法初始化一个新的叫做theAceOfSpades的常量:
let theAceOfSpades = BlackjackCard(rank: .Ace, suit: .Spades)
println("theAceOfSpades: \(theAceOfSpades.description)")
// prints "theAceOfSpades: suit is ♠, value is 1 or 11"
尽管Rank和Suit是嵌套在BlackjackCard之中的,但它们的类型可以从上下文中推断出来,所以这个实例的构造方法可以仅仅通过成员名字(.Ace和.Spades)引用枚举成员。上面的例子中,description属性正确的显示了纸牌黑桃A有一个值是1或者11.
引用嵌套的类型
为了在嵌套类型定义之外的上下文中使用,需要在嵌套类型的前面加上嵌套类型所在的类型名称:
let heartsSymbol = BlackjackCard.Suit.Hearts.rawValue
// heartsSymbol is "♡"
对于上例,这使得Suit、Rank和Values的名字简短,因为它们的名字自然和定义的上下文一致。