美文网首页Swift
Swift教程之枚举

Swift教程之枚举

作者: 暗夜夜夜行路 | 来源:发表于2017-10-04 17:44 被阅读7次

    枚举

    C语言中,C枚举将相关名称分配给一组整数值,而Swift枚举不必为每个枚举值提供一个值。若为每个枚举值提供了值(称为原始值),则该值可以是字符串、字符或任何整数或浮点类型的值。

    枚举还可以指定与每个枚举值一起存储的任何类型的关联值,这与其他语言中的联合或变体一样。

    Swift中的枚举拥有传统上仅由类支持的许多功能,例如计算属性来提供当前枚举值的附加信息,以及实例方法来提供与枚举值的相关功能。

    <br />

    枚举语法

    使用Enum关键字声明枚举:

    enum SomeEnumeration {
        // enumeration definition goes here
    }
    

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

    enum CompassPoint {
        case north
        case south
        case east
        case west
    }
    

    使用case关键字引入新的枚举情况。

    注意

    与C和Objective-C不同,Swift的枚举值不会在创建时分配默认整数值。上述枚举值不会隐式地等于0、1、2、3。

    多个case可以写在一行,用逗号隔开:

    enum Planet {
        case mercury, venus, earth, mars, jupiter, saturn, uranus, neptune
    }
    

    每个枚举都是一个全新的类型,像其它类型一样,名称以大写字母开头,且为单数而非复数名称。

    var directionToHead = CompassPoint.west
    

    使用枚举类型将某个枚举值初始化时,Swift会推断其具体枚举类型,下次重新赋值时,可忽略枚举类型。使用点语法:

    directionToHead = .east
    

    <br />

    使用Switch语句匹配枚举值

    使用switch语句匹配单个枚举值:

    directionToHead = .south
    switch directionToHead {
    case .north:
        print("Lots of planets have a north")
    case .south:
        print("Watch out for penguins")
    case .east:
        print("Where the sun rises")
    case .west:
        print("Where the skies are blue")
    }
    // Prints "Watch out for penguins"
    

    switch语句必须包括所有没有枚举值,否则无法通过编译,确保某个枚举情况不会被意外省略。

    使用default涵盖未明确处理的任何枚举值:

    let somePlanet = Planet.earth
    switch somePlanet {
    case .earth:
        print("Mostly harmless")
    default:
        print("Not a safe place for humans")
    }
    // Prints "Mostly harmless"
    

    <br />

    关联值

    有时可以将其他类型的值和枚举值一起关联起来,与枚举值一起存储,并在每次使用该信息时,该信息会发生变化。

    以下为携带关联值的条形码枚举:

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

    关联值不提供类型的实际值,只提供枚举值存储的关联值的类型。

    使用一下任一类型创建枚举值:

    var productBarcode = Barcode.upc(8, 85909, 51226, 3)
    
    productBarcode = .qrCode("ABCDEFGHIJKLMNOP")
    

    使用switch提取关联值:

    switch productBarcode {
    case .upc(let numberSystem, let manufacturer, let product, let check):
        print("UPC: \(numberSystem), \(manufacturer), \(product), \(check).")
    case .qrCode(let productCode):
        print("QR code: \(productCode).")
    }
    // Prints "QR code: ABCDEFGHIJKLMNOP."
    

    关联值全被提取为常量或变量:

    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."
    

    <br />

    原始值

    枚举值可以使用类型一致的默认值(即原始值)填充。

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

    原始值可以是字符串、字符或任何整数或浮点数类型,每个原始值必须是唯一的。

    隐式分配枚举值

    不必为每个枚举值显示分配原始值,Swift会自动分配。

    整数用于原始值时,第一个枚举值自动设置为0,之后递增1。

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

    上面例子中,Planet.Planet具有显式原始值1,其他为隐式枚举值是在其基础上递增1。

    字符串作为原始值时,每个枚举值的隐含原始值是该枚举值的字符串字面量。

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

    上面例子中,CompassPoint.south的隐式原始值为“south”,以此类推。

    使用枚举类型的rawValue属性获取其枚举值的原始值:

    let earthsOrder = Planet.earth.rawValue
    // earthsOrder is 3
     
    let sunsetDirection = CompassPoint.west.rawValue
    // sunsetDirection is "west"
    

    用原始值初始化枚举值

    若使用原始值类型定义枚举,枚举会自动创建一个初始化器,并可通过原始值的初始化器创建枚举的实例。

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

    然而,并不是所有的原始值都能找到匹配的枚举值,所以原始值初始化器(可失败的初始化器)总是返回枚举的可选类型,可结合可选绑定和switch语句处理该可选类型。

    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"
    

    <br />

    递归枚举

    当某个枚举值的关联值类型是该枚举类型时,该枚举是递归枚举。

    递归枚举有两种书写方式:

    1. 在需要递归的枚举值前加上indirect关键字,给部分枚举值启用递归。

    `

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

    2.在声明枚举类型enum关键字前加上indirect关键字,给所有枚举值启用递归。

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

    创建上述递归枚举实例:

    let five = ArithmeticExpression.number(5)
    let four = ArithmeticExpression.number(4)
    let sum = ArithmeticExpression.addition(five, four)
    let product = ArithmeticExpression.multiplication(sum, ArithmeticExpression.number(2))
    

    将递归枚举封装成递归函数:

    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"
    

    相关文章

      网友评论

        本文标题: Swift教程之枚举

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