Files
the-swift-programming-langu…/source/chapter2/08_Enumerations.md
2014-06-08 14:54:31 +02:00

8.1 KiB
Raw Blame History

枚举

本页内容包含:

  • 枚举语法
  • 匹配枚举值与Swith语句
  • 实例值
  • 原始值

枚举enumeration定义了一个通用类型的一组相关的值使你可以在你的代码中以一个安全的方式来使用这些值。

如果你熟悉C语言你就会知道在C语言中枚举会把一些整型值赋予枚举标识符。Swift中的枚举更加灵活不需要给每一个枚举成员提供一个值。如果一个值一个“原始”值被提供给每个枚举成员enumeration member则该值可以是一个字符串一个字符或是一个整型值或浮点值。

此外枚举成员可以指定任何类型的实例值associated value被每个枚举成

在Swift中枚举类型是一等公民类型。它们采用了很多传统上只被类class)所支持的特征例如计算属性computed properties)用于提供关于枚举当前值的附加信息 实例方法instance methods用于提供和枚举所代表的值相关联的功能。枚举也可以定义实例初始化initializers来提供一个初始成员值可以在原始的实现基础上扩展它们的功能可以遵守协议protocols来提供标准的功能。

欲了解更多相关功能请参见属性Properties方法Methods构造过程Initialization扩展Extensions和接口Protocols

枚举语法

使用enum关键词并且把它们的整个定义放在一对大括号内:

enum SomeEumeration {
	// enumeration definition goes here
}

以下是一个指南针四个方向的例子:

enum CompassPoint {
	case North
	case South
	case East
	case West
}

在一个枚举中定义的值(例如 NorthSouthEastWest)是枚举的成员值(或者成员)。case关键词表明新的一行成员值将被定义。

注意: 不像C和Objective-C一样Swift的枚举成员在被创建时不会被赋予一个默认的整数值。在上面的CompassPoints例子中,NorthSouthEastWest不是隐示得等于0123。相反的,这些不同的枚举成员在CompassPoint的一种显示定义中拥有各自不同的值。

多个成员值可以出现在同一行上,用逗号隔开:

enum Planet {
	case Mercury, Venus, Earth, Mars, Jupiter, Saturn, Uranus, Nepturn
}

每个枚举定义了一个全新的类型。像Swift中其他类型一样它们的名字例如CompassPointPlanet)必须以一个大写字母开头。给枚举类型单数名字而不是复数名字,以便于读起来更加容易理解:

var directionToHead = CompassPoint.West

directionToHead的类型被推断当它被CompassPoint的一个可能值初始化。一旦directionToHead被声明为一个CompassPoint,你可以使用更短的(.)语法将其设置为另一个CompassPoint的值:

directionToHead = .East

directionToHead的类型已知时,当设定它的值时,你可以不写类型名。使用显示类型的枚举值可以让代码具有更好的可读性。

匹配枚举值和Switch语句

你可以匹配单个枚举值和switch语句:

directionToHead = .South
switch directionToHead {
case .North:
	println("Lots of planets have a north")
case .South:
	println("Watch out for penguins")
case .East:
	println("Where the sun rises")
case .West:
	println("Where the skies are blue")
}
// prints "Watch out for penguins”

你可以如此理解这段代码:

“考虑directionToHead的值。当它等于.North,打印“Lots of planets have a north”。当它等于.South,打印“Watch out for penguins”。”

。。。。。。等等。

正如在控制流Control Flow中介绍当考虑一个枚举的成员们时一个switch语句必须全面。如果忽略了.West这种情况,上面那段代码将如果通过编译,因为它没有考虑到CompassPoint的全部成员。全面性的要求确保了枚举成员不会被意外遗漏。

当不需要匹配每个枚举成员的时候,你可以提供一个默认default分支来涵盖所有未明确提出的任何成员:

let somePlanet = Planet.Earth
switch somePlanet {
case .Earth:
	println("Mostly harmless")
default:
	println("Not a safe place for humans")
}
// prints "Mostly harmless”

实例值Associated Values

上一小节的例子演示了一个枚举的成员是如何被定义(分类)的。你可以为Planet.Earth设置一个常量或则变量,并且在之后查看这个值。然而,有时候会很有用如果能够把其他类型的实例值和成员值一起存储起来。这能让你随着成员值存储额外的自定义信息,并且当每次你在代码中利用该成员时允许这个信息产生变化。

你可以定义Swift的枚举存储任何类型的实例值如果需要的话每个成员的数据类型可以是各不相同的。枚举的这种特性跟其他语言中的可辨识联合discriminated unions标签联合tagged unions或者变体variants相似。

例如假设一个库存跟踪系统需要利用两种不同类型的条形码来跟踪商品。有些商品上标有UPC-A格式的一维码它使用数字0到9.每一个条形码都有一个代表“数字系统”的数字该数字后接10个代表“标识符”的数字。最后一个数字是“检查”位用来验证代码是否被正确扫描

其他商品上标有QR码格式的二维码它可以使用任何ISO8859-1字符并且可以编码一个最多拥有2,953字符的字符串。

对于库存跟踪系统来说能够把UPC-A码作为三个整型值的元组和把QR码作为一个任何长度的字符串存储起来是方便的。

在Swift中用来定义两种商品条码的枚举是这样子的

enum Barcode {
	case UPCA(Int, Int, Int)
	case QRCode(String)
}

以上代码可以这么理解:

“定义一个名为Barcode的枚举类型,它可以是UPCA的一个实例值(IntIntInt),或者QRCode的一个字符串类型(String)实例值。”

这个定义不提供任何IntString的实际值,它只是定义了,当Barcode常量和变量等于Barcode.UPCABarcode.QRCode时,其实例值的类型。

然后可以使用任何一种类型类创建新的条码,如:

var productBarcode = Barcode.UPCA(8, 85909_51226, 3)

以上例子创建了一个名为productBarcode的新变量,并且赋给它一个Barcode.UPCA的实例元组值(8, 8590951226, 3)。提供的“标识符”值在整数字中有一个下划线,使其便于阅读条形码。

同一个商品可以被分配一个不同类型的条形码,如:

productBarcode = .QRCode("ABCDEFGHIJKLMNOP")

这时,原始的Barcode.UPCA和其整数值被新的Barcode.QRCode和其字符串值所替代。条形码的常量和变量可以存储一个.UPCA或者一个.QRCode(连同其实例值),但是在任何指定时间只能存储其中之一。

像以前那样不同的条形码类型可以使用一个switch语句来检查然而这次实例值可以被提取作为switch语句的一部分。你可以在switch的case分支代码中提取每个实例值作为一个常量let前缀)或者作为一个变量(用var前缀)来使用:

switch productBarcode {
case .UPCA(let numberSystem, let identifier, let check):
	println("UPC-A with value of \(numberSystem), \(identifier), \(check).")
case .QRCode(let productCode):
	println("QR code with value of \(productCode).")
}
// prints "QR code with value of ABCDEFGHIJKLMNOP.”

如果一个枚举成员的所有实例值被提取为常量,或者它们全部被提取为变量,为了简洁,你可以只放置一个var或者let标注在成员名称前:

switch productBarcode {
case let .UPCA(numberSystem, identifier, check):
	println("UPC-A with value of \(numberSystem), \(identifier), \(check).")
case let .QRCode(productCode):
	println("QR code with value of \(productCode).")
}
// prints "QR code with value of ABCDEFGHIJKLMNOP."

原始值Raw Values