一、基本使用

先看枚举的几种使用(暂不要问,看看是否都能看懂,待会会逐一讲解)

1、操作一 简单使用

//第一种方式
enum Direction {
case east
case west
case south
case north func testDir() -> String {
switch self {
case .east:
return "东边"
case .west:
return "西边"
case .south:
return "南边"
case .north:
return "北边"
}
}
} //第二种方式
enum Direction1 {
case east, west, south, north func testDir() -> String {
switch self {
case .east:
return "东边"
case .west:
return "西边"
case .south:
return "南边"
case .north:
return "北边"
}
}
} var dir = Direction.east
dir = .north
var dir1 = Direction1.east
dir1 = .north

第一种和第二种完全一样。

2、操作二 关联值(Associated Values)

关联值(Associated Values)将枚举的成员值跟其他类型的值关联存储在一起,非常有用!

2.1 关联值示例1

enum Score {
case points(Int)
case grade(Character)
}
var score = Score.points()
score = .grade("A") func testScore() {
switch score {
case let .points(i):
print(i,"points")
case let .grade(i):
print("grade",i)
}
}

2.2 关联值示例2

enum Date {
case digit(year: Int, month: Int, day: Int)
case string(String)
} var date = Date.digit(year: , month: , day: )
date = .string("2011-09-10") func testDate() {
switch date {
case .digit(let year, let month, let day):
print(year, month, day)
case let .string(value):
print(value)
}
}

2.3 关联值示例3

手机密码方式有以下两种,可以用枚举表示,用关联值表示,图如下:

上面的图可以用关联值枚举表示如下

enum Password {
case number(Int, Int, Int, Int)
case gesture(String)
} var pwd = Password.number(, , , )
pwd = .gesture("") func testPwd() {
switch pwd {
case let .number(n1, n2, n3, n4):
print("number is ", n1, n2, n3, n4)
case let .gesture(str):
print("gesture is " ,str)
}
}

3、操作三 原始值(Raw Values)

原始值: 枚举成员可以使用相同类型的默认值预先对应,默认值叫做原始值,原始值不占用枚举变量的内存

enum PokerSuit: Character {
case spade = "黑"
case heart = "红"
case diamond = "方"
case club = "花"
} var suit = PokerSuit.spade
print(suit.rawValue)//黑
print(PokerSuit.club.rawValue)//花

4、隐式原始值(Implicitly Assigned Raw Values)

如果枚举的原始值是Int,String,Swift会自动分配原始值,和成员值一样

4.1 隐式原始值示例1

Direction和Direction1意义是一样的

enum Direction: String {
case north = "north"
case south = "south"
case east = "east"
case west = "west"
} enum Direction1: String {
case north, south, east, west
} print(Direction.north) //north
print(Direction1.north.rawValue)// north

4.2 隐式原始值示例2

Int类型,默认为0开始

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

4.3 隐式原始值示例3

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

上面讲述枚举的基本使用,下面我们将进入核心内容-从汇编的角度来看枚举的内存!!

二、汇编角度看枚举内存

示例1: 简单实用-通过下面代码查看枚举实例占用多少内存字节等

enum TestEnum {
case test1, test2, test3
} var t = TestEnum.test1
t = .test2
t = .test3

通过MemoryLayout查看内存大小

enum TestEnum {
case test1, test2, test3
} var t = TestEnum.test1
t = .test2
t = .test3
print(MemoryLayout<TestEnum>.size) //TestEnum实际占用内存空间
print(MemoryLayout<TestEnum>.stride)//系统分配给TestEnum的内存空间
print(MemoryLayout<TestEnum>.alignment)//对齐参数

运行结果如下

其实Swift还是很聪明的,仅仅使用一个字节来判断对象的不同,下面窥探test1,test2,test3的内存

因为Swift不支持枚举看底层的,所以通过一个内存访问小工具查看内存地址,然后通过内存地址查看内存布局

拿到内存地址后,可以通过view memory查看内容

将地址0x0000000100006660输入进去

因为通过上面发现占用一个字节,所以看第一个字节存储的为00,t为test1时

将断点向后移,看t = test2时,t的内存存储的时

再次看下t = test2 内存存储的值为

最后看下t = test3内存存储为

这种形式的枚举定义形式占用一个字节,可以代表的枚举范围也就是0x00-0xFF共256个case,足以表示所有情况的枚举穷举啦!

从示例1中,当枚举里面仅仅是case多个对象,枚举内存仅仅会分配1个字节来存储各个case,case对应的为0,1,2……

示例2 带有原始值

enum TestEnum: Int {
case test1 = , test2 = , test3 =
}
var t = TestEnum.test1
t = .test2
t = .test3

观察上面带有原始值枚举分配内存和占用内存情况

从最上面讲述带有原始值的枚举(红色标记)原始值不占用枚举变量的内存

所以仅仅需要1个字节来区分test1, test2,test3,我们再来看一个test2,看内存存储的是多少

看出test2存储的是依然是1,和原始值内容没有任何关系,存储和示例1没有区别,再次印证了,原始值不占用枚举变量的内存,不影响枚举内存结构和存储

示例3 带有关联值的枚举内存结构

关联值从上面基本使用得出关联值(Associated Values)将枚举的成员值跟其他类型的值关联存储在一起

enum TestEnum {
case test1(Int, Int, Int)
case test2(Int, Int)
case test3(Int)
case test4(Bool)
case test5
}
var t = TestEnum.test1(, , )
t = .test2(, )
t = .test3()
t = .test4(true)
t = .test5

继续使用MemoryLayout来看内存分配

从上面可看出TestEnum枚举实际占用内存空间大小为25,又因为内存对齐为8,所以系统分配了32个字节的大小给TestEnum

下面着重讲解为什么实际占用了25个字节,又是怎么存储的?

通过内存小工具查看枚举地址

然后View Memory工具查看内存结构如下

上面得出Int占据8个字节,对于TestEnum.test1(1, 2, 3)用24个字节存储这些关联值,得出关联值(Associated Values)将枚举的成员值跟其他类型的值关联存储在一的结论是正确的!

(拓展:为什么01,02放在前面,为什么不是放在后面,这牵扯到大小端的问题?下面讲述)

下面看test2的存储结构t = .test2(4, 5)

看第25个字节为Test2为0x01 = 1, test1的第25个字节为0x00 = 0, 依次类推,查看test4应该为3,下面揭开谜底

关联值枚举存储结论

有一个字节存储成员值,用于区分哪一个成员值

N个字节存储关联值(N取占用内存量最大的关联值),任何一个case的关联值都会共用这N个字节

上面代码与查看内存小工具封装代码https://github.com/zxy1829760/SwiftEnum

 拓展-大小端问题

存储0x11223344,大小端存储如下

以上就是枚举内存的底层结构,希望对大家有所帮助!!! 下一篇将讲述struct与class的区别!

Swift 枚举-从汇编角度看枚举内存结构的更多相关文章

  1. 从linux进程角度看JVM内存模型

    普通进程栈区,在JVM一般仅仅用做线程栈,如下图所示 首先是永久代.永久代本质上是Java程序的代码区和数据区.Java程序中类(class),会被加载到整个区域的不同数据结构中去,包括常量池.域.方 ...

  2. Swift 里 Array (一)内存结构

    public struct Array<Element>: _DestructorSafeContainer { #if _runtime(_ObjC) @usableFromInline ...

  3. c++ 汇编代码看内存分配

    汇编代码看内存分配 (1). 程序运行时分为存储区域分为 存储区域 存储内容 extra 代码区 存放代码指令,包括除字符串常量的字面值 静态存储区 存放静态变量和全局变量 执行main之前就分配好了 ...

  4. Swift--struct与class的区别(汇编角度底层分析)

    概述 相对Objective-C, Swift使用结构体Struct的比例大大增加了,其中Int, Bool,以及String,Array等底层全部使用Struct来定义!在Swift中结构体不仅可以 ...

  5. swift_枚举 | 可为空类型 | 枚举关联值 | 枚举递归 | 树的概念

    ***************可为空的类型 var demo2 :we_demo = nil 上面这个代码串的语法是错的 为什么呢, 在Swift中,所有的类型定义出来的属性的默认值都不可以是nil ...

  6. java枚举与.net中的枚举区别

    通过一段时间的项目实践,发现java中的枚举与.net中的枚举有很大的差别,初期造成了我对java中的枚举一些错误理解及部分有缺陷的应用,其实追其原因还是因为我会习惯性的认为java的枚举在作用以及定 ...

  7. 从free命令看Linux内存管理

    free命令是Linux系统下用来查看内存使用情况的,例如: $ free -h total used free shared buffers cached Mem: 7.8G 6.6G 1.3G 0 ...

  8. 从线程模型的角度看Netty的高性能

    转载:Netty(二) 从线程模型的角度看 Netty 为什么是高性能的? 传统 IO 在 Netty 以及 NIO 出现之前,我们写 IO 应用其实用的都是用 java.io.* 下所提供的包. 比 ...

  9. 从JDK源码角度看Object

    Java的Object是所有其他类的父类,从继承的层次来看它就是最顶层根,所以它也是唯一一个没有父类的类.它包含了对象常用的一些方法,比如getClass.hashCode.equals.clone. ...

随机推荐

  1. Vmware Ubuntu18.04更换清华源

    一.安装Ubuntu18.04 省略 二.安装VmwareTool 1.选择机器右击安装2.打开文件,copy压缩文件到其它目录(理由: 内存不够解压)3.解压文件,运行./忘记名字了.pl文件4.注 ...

  2. flash-session

    作用:更改session存储的位置 1.session默认存放在浏览器的cookie中 源码 wsgi->app.__call__->wsgi_app->push->self. ...

  3. Postman post csrf_token

    1.填入代码 var csrf_token = postman.getResponseCookie("csrftoken").value postman.clearGlobalVa ...

  4. SqlServer分页存储过程(多表查询,多条件排序),Repeater控件呈现数据以及分页

        存储过程(Stored Procedure)是在大型数据库系统中,一组为了完成特定功能的SQL 语句集,存储在数据库中,经过第一次编译后再次调用不需要再次编译,用户通过指定存储过程的名字并给出 ...

  5. 遗传编程GP-地图路径寻路

    本文介绍的是基于GP,并非A*算法,算是另类实现吧. 先看看地图定义,在文本文件中定义如下字符串,代表30列11行大小的地图 初始位置在左上角(0,0) ,值为1的是允许走的通的路,目标位置为右下角( ...

  6. 美食家app开发日记

    民以食为天. 作为一个20年几乎没做过饭的吃货,从这个假期开始,想利用些时间,自己动手尝试,做些好吃的出来,一方面给父母减轻点负担,获得点成就感,一方面体验生活,学学厨艺,感受生活的乐趣和美好,其三, ...

  7. Idea破解至2089年

    我是用的版本是2018.3.6,别的朋友使用的是2019的某个版本,不过关都不影响破解 下载jar包:链接:https://pan.***baidu.***com/s/1aRR0***2YNI9jew ...

  8. 架构模式中的Active Record和Data Mapper

    架构模式中的Active Record和Data Mapper 概念 在简单应用中,领域模型是一种和数据库结构一致的简单结构,对应每个数据库表都有一个领域类,在这种情况下,有必要让每个对象负责数据库的 ...

  9. 每天一道Java题[8]

    以下题目及解答属于个人见解,欢迎大家也分享和补充一下解答的内容,互相促进,共同进步! 题目 RESTful WebService与SOAP WebService有什么异同? 解答 SOAP是一个协议, ...

  10. nginx的四个主要组成部分

    1.nginx二进制可执行文件 · 由各模块源码编译出的一个文件 2.nginx.conf配置文件 · 控制nginx的行为 3.access.log访问日志 . 记录每一条http请求信息 4.er ...