Swift 枚举-从汇编角度看枚举内存结构
一、基本使用
先看枚举的几种使用(暂不要问,看看是否都能看懂,待会会逐一讲解)
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 枚举-从汇编角度看枚举内存结构的更多相关文章
- 从linux进程角度看JVM内存模型
普通进程栈区,在JVM一般仅仅用做线程栈,如下图所示 首先是永久代.永久代本质上是Java程序的代码区和数据区.Java程序中类(class),会被加载到整个区域的不同数据结构中去,包括常量池.域.方 ...
- Swift 里 Array (一)内存结构
public struct Array<Element>: _DestructorSafeContainer { #if _runtime(_ObjC) @usableFromInline ...
- c++ 汇编代码看内存分配
汇编代码看内存分配 (1). 程序运行时分为存储区域分为 存储区域 存储内容 extra 代码区 存放代码指令,包括除字符串常量的字面值 静态存储区 存放静态变量和全局变量 执行main之前就分配好了 ...
- Swift--struct与class的区别(汇编角度底层分析)
概述 相对Objective-C, Swift使用结构体Struct的比例大大增加了,其中Int, Bool,以及String,Array等底层全部使用Struct来定义!在Swift中结构体不仅可以 ...
- swift_枚举 | 可为空类型 | 枚举关联值 | 枚举递归 | 树的概念
***************可为空的类型 var demo2 :we_demo = nil 上面这个代码串的语法是错的 为什么呢, 在Swift中,所有的类型定义出来的属性的默认值都不可以是nil ...
- java枚举与.net中的枚举区别
通过一段时间的项目实践,发现java中的枚举与.net中的枚举有很大的差别,初期造成了我对java中的枚举一些错误理解及部分有缺陷的应用,其实追其原因还是因为我会习惯性的认为java的枚举在作用以及定 ...
- 从free命令看Linux内存管理
free命令是Linux系统下用来查看内存使用情况的,例如: $ free -h total used free shared buffers cached Mem: 7.8G 6.6G 1.3G 0 ...
- 从线程模型的角度看Netty的高性能
转载:Netty(二) 从线程模型的角度看 Netty 为什么是高性能的? 传统 IO 在 Netty 以及 NIO 出现之前,我们写 IO 应用其实用的都是用 java.io.* 下所提供的包. 比 ...
- 从JDK源码角度看Object
Java的Object是所有其他类的父类,从继承的层次来看它就是最顶层根,所以它也是唯一一个没有父类的类.它包含了对象常用的一些方法,比如getClass.hashCode.equals.clone. ...
随机推荐
- 关于neo4j初入门(2)
DELETE删除 删除节点及相关节点和关系. DELETE <node-name-list> DELETE <node1-name>,<node2-name>,&l ...
- 实验二:在Cisco Packet Tracer模拟器上进行Trunk+Access端口混合模式实验
1.配置图 2.配置命令 Switch0的VLAN配置如下: 查看Switch0的vlan配置如下: Switch0的Trunk端口配置如下: Switch1的VLAN配置如下: 查看Switch1的 ...
- CAP原理和BASE理论
CAP原理 概述 CAP理论的主要场景是在分布式环境下,在单机环境下,基本可不考虑CAP问题. CAP理论就是说在分布式存储系统中,最多只能实现上面的两点.而由于当前的网络硬件肯定会出现延迟丢包等问题 ...
- 浅析YYCache
一.前言 读优秀的源码,对自己的提升还是很快的,无论是考虑问题的角度,还是编码能力. 带着问题读源码的,学习效率更高,可以暂时先定几个小问题,带着问题,去思考为什么作者这样弄,是否有替换方案? 1). ...
- 使用Qt自动注册Lav
Qt播放视频使用QMediaPlayer要注册Lav解码器,如果手动去注册,每次去使用管理员运行命令或者生成.bat文件都比较麻烦. 解决方法步骤如下: 一:编写注册Lav解码器脚本,并取消控制台的显 ...
- centos7 配置虚拟交换机(物理交换机truck端口设置)(使用brctl)
转自:http://blog.csdn.net/qq_21398167/article/details/46409503 虚拟交换机配置 inux VLAN配置(vconfig) 安装vlan(vco ...
- jmeter性能测试2:基础功能介绍
对于英语不好的同学建议先改为简体中文再进行使用 1.添加->threads->线程组(控制总体并发) 线程数:虚拟用户数.一个虚拟用户占用一个进程或线程 ...
- scrapy-redis分布式爬虫实战
Scrapy-Redis代码实战 Scrapy 是一个通用的爬虫框架,但是不支持分布式,Scrapy-redis是为了更方便地实现Scrapy分布式爬取,而提供了一些以redis为基础的组件(仅有组件 ...
- winsocket入门学习
参考资料:http://c.biancheng.net/cpp/socket/ http://www.winsocketdotnetworkprogramming.com/ socket 是" ...
- 牛客练习赛25 A 因数个数和(数论分块)
题意: q次询问,每次给一个x,问1到x的因数个数的和. 1<=q<=10 ,1<= x<=10^9 1s 思路: 对1~n中的每个数i,i作为i,2i,3i,...的约数,一 ...