Swift3知识点梳理(一):枚举

在许多编程语言中都存在的枚举类型,基本都是用于定义一个具有一组数据的值,且这些值是可以枚举的。那么Swift中我们该如何定义并使用枚举类型呢?我们从以下3个方面对Swift3中的枚举进行梳理。

Swift3中枚举的3种形式


1. 无raw value且无associated value
(1)定义无raw value且无associated value的枚举
这种枚举类型的定义非常简单,例如:

enum SwitchState {
case ON 
case OFF
}

它定义了开关的两种状态: ON, OFF。另外,它还有一种更简洁的定义方式:

enum SwitchState {
case ON, OFF
}

(2)初始化及使用无raw value且无associated value类型的枚举
示例:

var state = SWitchState.ON
switch state {
case .ON: print("This is ON state.")
case .OFF: print("This is OFF state.")
}

2. 有raw value
它用于给枚举中定义的各个对象预先定义一个值。
(1)定义有raw value的枚举

enum ASCIIControlCharacter: Character {
case tab = "\\t"
case lineFeed = "\\n"
case carriageReturn = "\\r"
}

它定义了一个包含一系列ASCII字符的枚举类型,同时给其中的每个对象预先定义了一个字符串类型的值。如果预先定义的值的类型一致,那么我们可以用更简洁的方式定义,例如:

enum CompassPoint: String {
case north, south, east, west
}

如果预先定义的值是Int类型,且值是递增的,那么我们可以用如下方式定义:

enum Planet: Int {
case mercury = 1, venus, earth, mars, jupiter, saturn, uranus, neptune
}

(2)使用raw value 初始化有raw value的枚举

let possiblePlanet = Planet(rawValue: 7)
// possiblePlanet is of type Planet? and equals Planet.uranus

(3)使用有raw value的枚举的方式

let positionToFind = 11
if let somePlanet = Planet(rawValue: positionToFind) {
     switch somePlanet {
     case .earth:
        print("Mostly harmless")
     default:
        print("Not a safe place for humans")
     }
} else {
    print("There isn't a planet at position \\(positionToFind)")
}
// Prints "There isn't a planet at position 11"

关于raw value需要注意的地方如下:

  • 枚举中定义的各值所对应的raw value均是常量,不可变化且互不相同;
  • raw value对应的数据类型只能是String, Character, Int 或者Float类型。

3. 有associated value
在定义枚举的时候注入一个与枚举值相关联的数据,这个数据以构造方法的形式注入到对象中。
(1)定义有associated value的枚举

enum Barcode {
case upc(Int, Int, Int, Int)
case qrCode(String)
}

(2)初始化及使用有associated value的方式

var productBarcode = Barcode.upc(8, 85909, 51226, 3)
switch productBarcode {
case let .upc(numberSystem, manufacturer, product, check):
    print("UPC : \\(numberSystem), \\(manufacturer), \\(product), \\(check).")
case let .qrCode(productCode):
    print("QR code: \\(productCode).")
}
// Prints "QR code: ABCDEFGHIJKLMNOP."

关于Associated value需要注意如下两点:

  • 关联的数据变量可以是任意类型,且枚举内定义的值的类型可以不同;
  • 数据是变量而非常量。

Swift3中2个特殊的枚举类型


1. Optional
(1)Optional的枚举本质
要定义一个Optional类型的变量有两种方式, 第一种方式,
let userName: Optional
第二种方式是第一种方式的语法糖形式,这也是最常用的定义optional的方式:
let studentName: String?
剥开事物看本质,对于Optional其实质就是枚举类型。看一下苹果官方文档对于optional的定义:

public enum Optional : _Reflectable, NilLiteralConvertible {
case None
case Some(Wrapped)
...
}

根据定义,我们可以看到Optional的本质就是枚举类型。
(2)空虚的nil
nil是Optional枚举实例的一个值,即上段代码中的'None'值。nil有别于其它语言中的null值,以及OC中的nil值,它仅仅表示的是该对象不包含值,而并非指向一个不存在的对象。

关于nil的几个注意点:

  • 可以给任意类型的Optional对象设置为nil,而不仅仅是class类型;
  • Optional对象的默认值为nil;
  • 不能给非Optional类型的变量赋值为nil

(3)?与!兄弟
'?'和'!'兄弟是一对非常有趣的修饰符,用途非常普遍。在swift开发中,你可以经常看到它们兄弟在代码中出没的身影,可以说哪哪都是。为什么说它们有趣呢?因为它们有着非常奇妙的功能,对于Optional类型的使用非常关键。'?'和'!'都可以用于变量声明以及表达式中,但是有着各自的作用。

  • 用途一:定义optional变量
    '?'定义的变量是optional类型,如:let studentName: String?
    '!'定义的变量也是optional类型,如:let studentName: String! , 然而与'?'定义的变量不同的是,'?'定义的变量在使用时需要使用'!'进行解包,而使用'!'定义的变量无需解包就可以使用。总之,'!'定义变量的方式可以看成是使用'?'定义变量并使用'!'解包的快捷方式。需要注意的是我们应确保使用'!'定义的变量时刻有具体的值,否则使用时会造成crash。

  • 用途二:用于表达式中解包
    '!'作用于表达时中时主要用于解包,例如:

class Person {
    var residence: Residence?
}
class Residence {
    var numberOfRooms = 1
}
let john = Person()
let roomCount = john.residence!.numberOfRooms
// this triggers a runtime error

'?'主要用于optional chain表达中,作用也是解包。如果当前值为nil时表达时返回nil,且不会造成crash,例如:

class Person {
    var residence: Residence?
}
class Residence {
    var numberOfRooms = 1
}
let john = Person()
//let roomCount = john.residence!.numberOfRooms
// this triggers a runtime error, the right way is as the following:
if let roomCount = john.residence?.numberOfRooms {
    print("John's residence has \\(roomCount) room(s).")
} else {
    print("Unable to retrieve the number of rooms.")
}
// Prints "Unable to retrieve the number of rooms."

(4)解包的三种方式

  • 解包的方法一:'!'运算符
    例如:
var userName: String? = "Mike"
print("The user name is \\(userName!)")
  • 解包的方法二:optional binding: if let {}
    例如:
var userName: String? = "Mike"
if let name = userName {
    print("The user name is \\(name)")
}
  • 解包的方法三:optional chain:
    如前所述,我们可以使用optional chain进行解包,这里不再具体阐述。

2. 迭代枚举
迭代枚举其实是具有associated value类型枚举的一个特例。迭代枚举是一个非常有意思的语法,类似于函数的嵌套迭代,它允许在枚举定义Associated value的时候,将其值定义为与当前枚举相同的数据类型,从而形成了一种迭代的特征。定义如下:

enum ArithmeticExpression {
case number(Int)
indirect case addition(ArithmeticExpression, ArithmeticExpression)
indirect case multiplication(ArithmeticExpression, ArithmeticExpression)
}

迭代枚举的使用方式如下:

func evaluate(_ expression: ArithmeticExpression) -> Int {
    switch expression {
    case let .number(value):
        return value
    case let .addition(left, right):
        return evaluate(left) + evaluate(right)
    case let .multiplication(left, right):
        return evaluate(left) * evaluate(right)
    }
}
print(evaluate(product))
// Prints "18"

Swift3中枚举的面向对象


枚举类型在swift中属于一等类型,它和class以及struct共同组成swift的面向对象的三大数据类型。枚举类型虽然是值类型,但是它还是具有很多面向对象的特征:

  • 有构造方法,可以通过使用构造方法初始化。
  • 可以定义computed属性;
  • 可以定义实例方法;
  • 可以实现接口;
  • 可以被扩展。

鉴于篇幅与内容的原因,具体的内容将会在之后的swift面向对象的各个专题中分别总结。下一篇文章将会总结元组类型,即:Swift3知识梳理(二):元组类型。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容

  • importUIKit classViewController:UITabBarController{ enumD...
    明哥_Young阅读 3,784评论 1 10
  • title: "Swift 中枚举高级用法及实践"date: 2015-11-20tags: [APPVENTUR...
    guoshengboy阅读 2,565评论 0 2
  • 出自乡野中, 往来于闹市。 为练人情学世故, 万般未洞明。 转眼三十已过, 经年又逢不惑。 吃得苦,做得乐, 心事...
    绘享瘦阅读 357评论 1 1
  • 茉莉花茶:驱寒 蒲公英茶 枸杞茶:滋肾、养肝、润肺、明目 柠檬草:改善贫血、调和肠胃、调节油脂分泌和降低血脂、缓解...
    践行学堂阅读 257评论 0 0