[翻译]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结构定义了两个属性来表现这些: 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的名字简短,因为它们的名字自然和定义的上下文一致。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |