美文网首页iOS开发互联网科技iOS Developer
Swift3知识点梳理(一):枚举

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

作者: Ifdef_Max | 来源:发表于2016-09-24 13:17 被阅读198次

    在许多编程语言中都存在的枚举类型,基本都是用于定义一个具有一组数据的值,且这些值是可以枚举的。那么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知识梳理(二):元组类型。

    相关文章

      网友评论

        本文标题:Swift3知识点梳理(一):枚举

        本文链接:https://www.haomeiwen.com/subject/dkityttx.html