Swift之枚举

为了表示我对『饱醉豚』事件的不满,不再更新,后续有文章只更新 个人博客和 掘金

欢迎移步 个人博客
或者 掘金

本文首发于我的个人博客

枚举

枚举的基本用法

定义

//定义方向的枚举
enum Direction {
     case north
     case south
     case east
     case west 
}

上面也可以写成

enum Direction {
    case north, south, east, west
}

使用

var dir = Direction.west 
dir = Direction.east 
dir = .north
print(dir) // north

也可以在switch中使用


switch dir { 
case .north:
    print("north") 
case .south:
    print("south") 
case .east:
    print("east") 
case .west:
    print("west")
}

关联值

有时将枚举的成员值跟其他类型的值关联存储在一起,会非常有用,可以认为将值直接存入到枚举的内存中


  enum Score {
    case points(Int)
    case grade(Character)
}

如下使用:


var score = Score.points(88) 
score = .grade("A")

如果我们想使用枚举的具体值,可以如下用 i 来保存数据


  switch score {
    case let .points(i):
        print(i, "points") 
    case let .grade(i):
        print("grade", i)
} // grade A

再比如我们想定义日期的枚举值,可以如下:


enum Date {
    case digit(year: Int, month: Int, day: Int)
    case string(String)
}

//使用的时候,可以直接传年月日,或者传字符串

var date = Date.digit(year: 2011, month: 9, day: 10) 
date = .string("2011-09-10")
switch date {
    case .digit(let year, let month, let day):
            rint(year, month, day) 
    case let .string(value):
        print(value)
}

必要时let也可以改为var

原始值

枚举成员可以使用相同类型的默认值预先对应,这个默认值叫做:原始值


// 定义枚举
enum Grade : String {
    case perfect = "A"
    case great = "B"
    case good = "C"
    case bad = "D"
}

// 使用
print(Grade.perfect.rawValue) // A 
print(Grade.great.rawValue) // B 
print(Grade.good.rawValue) // C 
print(Grade.bad.rawValue) // D

注意:原始值不占用枚举变量的内存

隐式原始值(Implicitly Assigned Raw Values)

如果枚举的原始值类型是Int、String,Swift会自动分配原始值

原始值是 String 类型枚举值


// 定义枚举值
  enum Direction : String {
    case north = "north"
    case south = "south"
    case east = "east"
    case west = "west"
}
// 等价于
enum Direction : String {
    case north, south, east, west
}
// 使用
print(Direction.north) // north 
print(Direction.north.rawValue) // north

原始值是 Int 类型枚举值


enum Season : Int {
    case spring, summer, autumn, winter
}
print(Season.spring.rawValue) // 0 
print(Season.summer.rawValue) // 1 
print(Season.autumn.rawValue) // 2 
print(Season.winter.rawValue) // 3

如果自己指定了原始值


enum Season : Int {
    case spring = 1, summer, autumn = 8, winter
}
print(Season.spring.rawValue) // 1 
print(Season.summer.rawValue) // 2 
print(Season.autumn.rawValue) // 8 
print(Season.winter.rawValue) // 9

递归枚举

递归枚举要加上关键字 indirect

eg:

// 递归枚举
indirect enum ArithExpr {
    case number(Int)
    case sum(ArithExpr, ArithExpr)
    case difference(ArithExpr, ArithExpr)
}

// 上面的递归枚举和下面的等效

enum ArithExpr {
    case number(Int)
    indirect case sum(ArithExpr, ArithExpr) 
    indirect case difference(ArithExpr, ArithExpr)
}

// 下列几种使用枚举都是可以的

let five = ArithExpr.number(5)
let four = ArithExpr.number(4)
let two = ArithExpr.number(2)
let sum = ArithExpr.sum(five, four)
let difference = ArithExpr.difference(sum, two)


// 自己写个 calculate 方法,
func calculate(_ expr: ArithExpr) -> Int { 
    switch expr {
    case let .number(value): 
        return value
    case let .sum(left, right):
        return calculate(left) + calculate(right)
    case let .difference(left, right):
        return calculate(left) - calculate(right)
} }

//最终调用,计算差值
calculate(difference)

MemoryLayout

可以使用MemoryLayout获取数据类型占用的内存大小

关联值

  • 将关联值直接存入到枚举内存中
// 定义枚举
 enum Password {
    case number(Int, Int, Int, Int)
    case other
}

MemoryLayout.stride // 40, 分配占用的空间大小 
MemoryLayout.size // 33, 实际用到的空间大小 4*8 + 1 = 33
MemoryLayout.alignment // 8, 对齐参数

定义变量来使用


var pwd = Password.number(9, 8, 6, 4)

MemoryLayout.stride(ofValue: pwd) // 40
MemoryLayout.size(ofValue: pwd) // 33 
MemoryLayout.alignment(ofValue: pwd) // 8
 
// 如果改变了pwd
 pwd = .other 
MemoryLayout.stride(ofValue: pwd) // 40
MemoryLayout.size(ofValue: pwd) // 33 
MemoryLayout.alignment(ofValue: pwd) // 8

原始值

  • 原始值不会直接存入到枚举内存中
  • 如果是下面这种枚举,只需要1个字节就可以了
  • 一个字节可以存放 FF 也就是 0~255个枚举值。如果
enum Season : Int {
    case spring, summer, autumn = 8, winter
}

MemoryLayout.stride // 1, 分配占用的空间大小 
MemoryLayout.size // 1, 实际用到的空间大小 1
MemoryLayout.alignment // 1, 对齐参数

窥探内存

使用 窥探内存细节的小工具 我们可以很轻松的获取swift中,这些枚举值的内存地址

简单枚举内存


enum TestEnum{
    case k0,k1,k2,k3
}

var t = TestEnum.k1
print(Mems.ptr(ofVal: &t)) 

t = TestEnum.k2

执行完 print(Mems.ptr(ofVal: &t)) 代码之后
输出

0x00000001000054b8

此时去查看 0x00000001000054b8地址的数据,

 01 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00

执行完 t = TestEnum.k2 之后

此时去查看 0x00000001000054b8地址的数据,

 02 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00

具体查看内存

假设我们有这么下面带代码,考虑下内存怎么布局呢?


enum TestEnum {
        case test1(Int, Int, Int)
        case test2(Int, Int)
        case test3(Int)
        case test4(Bool)
        case test5
}

print(MemoryLayout.size)  // 25, 分配占用的空间大小
print(MemoryLayout.stride)    //32, 实际用到的空间大小
print(MemoryLayout.alignment)// 8, 对齐参数


var e = TestEnum.test1(1, 2, 3)
print(Mems.ptr(ofVal: &e))
  
  
e = .test2(4, 5)
print(Mems.memStr(ofVal: &e))
    
e = .test3(6)
     
e = .test4(true)
      
e = .test5

执行完之后,可知

TestEnum 这个占用内存为:

  • print(MemoryLayout.size) // 25, 分配占用的空间大小
  • print(MemoryLayout.stride) //32, 实际用到的空间大小
  • print(MemoryLayout.alignment)// 8, 对齐参数

具体内存里面存的是什么呢?可以借助上面说的 窥探内存细节的小工具 打印出来内存,然后利用Xcode的 view Memory 查看具体内存的值

结果如下

 enum TestEnum {
        case test1(Int, Int, Int)
        case test2(Int, Int)
        case test3(Int)
        case test4(Bool)
        case test5
    }
    
    
    print(MemoryLayout.size)  // 25, 分配占用的空间大小
    print(MemoryLayout.stride)    //32, 实际用到的空间大小
    print(MemoryLayout.alignment)// 8, 对齐参数

    // 1个字节存储成员值
    // N个字节存储关联值(N取占用内存最大的关联值),任何一个case的关联值都共用这N个字节
    // 共用体
    
    // 小端:高高低低
    // 01 00 00 00 00 00 00 00  //对应的TestEnum.test1传入的数值
    // 02 00 00 00 00 00 00 00
    // 03 00 00 00 00 00 00 00
    // 00                // TestEnum.test1在第0个位置
    // 00 00 00 00 00 00 00
    var e = TestEnum.test1(1, 2, 3)
    print(Mems.ptr(ofVal: &e))
    
    // 04 00 00 00 00 00 00 00
    // 05 00 00 00 00 00 00 00
    // 00 00 00 00 00 00 00 00
    // 01                        // TestEnum.test1在第1个位置
    // 00 00 00 00 00 00 00
    e = .test2(4, 5)
    print(Mems.memStr(ofVal: &e))
    
    // 06 00 00 00 00 00 00 00
    // 00 00 00 00 00 00 00 00
    // 00 00 00 00 00 00 00 00
    // 02
    // 00 00 00 00 00 00 00
    e = .test3(6)
    
    // 01 00 00 00 00 00 00 00
    // 00 00 00 00 00 00 00 00
    // 00 00 00 00 00 00 00 00
    // 03
    // 00 00 00 00 00 00 00
    e = .test4(true)
    
    // 00 00 00 00 00 00 00 00
    // 00 00 00 00 00 00 00 00
    // 00 00 00 00 00 00 00 00
    // 04
    // 00 00 00 00 00 00 00
    e = .test5

只有一个case

假如只有一个case,其占用的空间为0,不需要存值来区分是哪个case

enum TestEnum {
    case spring
}

print(MemoryLayout.size)      // 0, 分配占用的空间大小
print(MemoryLayout.stride)    //1, 实际用到的空间大小
print(MemoryLayout.alignment)// 1, 对齐参数

只有一个case,有关联值

enum TestEnum {
    case spring(Int)
}

print(MemoryLayout.size)      // 8, 分配占用的空间大小
print(MemoryLayout.stride)    //8, 实际用到的空间大小
print(MemoryLayout.alignment)// 8, 对齐参数

参考资料:

Swift官方源码

从入门到精通Swift编程

窥探内存细节的小工具

你可能感兴趣的:(Swift之枚举)