嵌套的类型
枚举经常被创建用来支撑特定类或者结构体的功能。类似的,为了在一个复合类型上下文中使用,定义纯粹的工具类和结构体也是可行的。为了做到这些,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的名字简短,因为它们的名字自然和定义的上下文一致。