[翻译]Swift编程语言—— 嵌套的类型

嵌套的类型

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值