Swift教程(八)--枚举

2019-07-31 10:00:17 浏览数 (1)

本次的教程是基于Swift5.1版本

枚举定义语法

首先,我们来看看在 swift 中定义枚举的语法:

代码语言:javascript复制
enum CompassPoint {
case north
case south
case east
case west
}

我们注意到,swift 的每个枚举项前面,都使用一个 case 关键字来标识。除了每行声明一个枚举项,也可以将这些枚举项放在一行中声明,每项之间用逗号分隔:

代码语言:javascript复制
enum CompassPoint {
case north,south, east, west
}

注意一点,Objective-C 和 C 语言那样, swift 中的枚举项不可以用 0,1,2 这样的数字值来代替。它们有自己的值。

枚举类型定义好之后,我们就可以将它的枚举值赋值给某个变量:

代码语言:javascript复制
var myNorth  = CompassPoint.north

并且,对于类型明确的变量,我们可以直接省去枚举的类型前缀:

代码语言:javascript复制
var mySouth:CompassPoint = .south

枚举的使用

枚举值可以在 switch 语句中进行匹配:

代码语言:javascript复制
switch myNorth {
case .north:
print("north")
case .south:
print("south")
case .east:
print("east")
case .west:
print("west")
}

在 switch 语句中的每个 case 中,我们提供各个枚举项的名称:.Monday,.Tuesday 等等。在 swift 中 switch 中匹配枚举项,必须显示的列举出所有的枚举项。也就是对于我们上面表示星期的枚举类型WeekDay, 我们对它的 switch 语句中必须将所有的枚举项分支都明确的写出来。否则就会有编译错误。

这个机制也体现了 Swift 类型安全的核心思想。如果我们觉得每个枚举项都要明确的指定行为比较麻烦,我们还可以使用 default 分支来对于其余的枚举项定义行为:

代码语言:javascript复制
switch myNorth {
case .north:
print("north")
case .south:
print("south")
case .east:
print("east")
default:
print("west")
}

总之,无论用 default 也好,还是明确对每一个枚举项指定行为也好,在 Swift 中,我们都必须对枚举类型下的每个值,指定确定的行为。不能漏掉其中任何一个可能性。

关联值(Associated Values)

在 Swift 中,我们还可以定义这样的枚举类型,它的每一个枚举项都有一个附加信息,来扩充这个枚举项的信息表示,这又叫做关联值。加入我们有一个枚举类型Shape 来表示形状。

这个形状可以是矩形,也可以是圆形,等等。而每种具体的形状又对应了不同的属性,比如矩形有长,宽,圆形有,圆心,半径,等等。那么枚举的关联值就可以帮我们解决这个问题:

代码语言:javascript复制
enum Shape {
case Rectangle(CGRect)
case Circle(CGPoint,Int)

}

我们看到,每个枚举项的后面,都包含了一对括号,这里面定义了这个枚举项的关联值的类型。对于 Rectangle 我们使用一个 CGRect 来表示他的原点和长宽属性。 而对于 Circle,我们使用一个包含了 CGPoint 和 Int 类型的元组(Tuple) 来表示这个圆的圆心和半径。

这样我们在初始化枚举类型的时候,我们就可以根据每个枚举项的关联值类型,为它指定附加信息了:

代码语言:javascript复制
var rect = Shape.Rectangle(CGRect(x: 1, y: 1, width: 100, height: 100))

var circle = Shape.Circle(CGPoint(x: 1, y: 1), 2)

我们再看一下,带有关联值的枚举项在 switch 语句中的用法:

代码语言:javascript复制
switch rect {
case .Rectangle(let rect):
print(rect)
case let .Circle(center,radius):
print(center,radius)
}
//输出:(1.0, 1.0, 100.0, 100.0)

我们在 case 后面用一对括号来输出枚举项的关联值,可以用 let 或者 var 关键字,分别作为常量和变量进行输出。我们这里这样来使用case .Rectangle(let rect)。对于关联值是包含多个值的元组类型的,我们可以将 let 关键字放置在枚举项类型的前面,这样就可以不用对每个关联值都声明let 关键字了,let .Circle(center, radius)。

原始值(Raw Values)

我们刚刚了解了关联值类型的枚举的使用,Swift 的枚举类型还提供了另外一个叫做原始值(Raw Values)的实现。和关联值不同,它为枚举项提供一个默认值,这个默认值是在编译的时候就确定的。而不像关联值那样,要再实例化枚举值的时候才能确定。

这也就是说,原始值对于同一个枚举项都是一样的。而关联值对于同一个枚举项只是值的类型相同,但具体的取值也是不同的。

下面我们来看一下定义枚举原始值 (Raw Values) 的方法:

代码语言:javascript复制
enum WeekDayWithRaw : String {
case Monday = "1. Monday"
case Tuesday = "2. Tuesday"
case Wednesday = "3. Wednesday"
case Thursday = "4. Thursday"
case Friday = "5. Friday"
case Saturday = "6. Saturday"
case Sunday = "7. Sunday"

}

还是表示星期的枚举类型,我们对每个枚举项都定义了一个默认的原始值,注意一下我们定义枚举的第一行代码,enum WeekDayWithRaw : String 我们在枚举定义的最后,多加了一个String 关键字,这就表示这个枚举的原始值(Raw Values) 是 String 类型的。

定义好了原始值后,我们就可以用枚举项的 rawValue 属性来输出它:

代码语言:javascript复制
print(WeekDayWithRaw.Saturday.rawValue)
//输出:6. Saturday

我们还可以通过原始值(Raw Values) 来初始化枚举类型:

代码语言:javascript复制
let day = WeekDayWithRaw(rawValue: "3. Wednesday")

这个初始化方法的返回值是一个 Optionals。所以我们可以用 Optionals 的组合链来使用它的返回值:

代码语言:javascript复制
if let day = WeekDayWithRaw(rawValue: "3. Wednesday") {
print(day)
}else{
print("error")
}

//输出:Wednesday

返回值为 Optionals 的类型,代表这个方法可以返回一个具体的值,也可以返回 nil, 因为我们传入初始化方法的原始值,可能会不等于我们预设的那几个值。

0 人点赞