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. ...
随机推荐
- Alibaba Nacos 服务发现组件集群部署
前面学习了单机模式下的启动,生产环境中部署nacos肯定是使用集群模式cluster保证高可用. 官方文档的集群部署推荐使用VIP+域名模式,把所有服务列表放到一个vip下面,然后挂到一个域名下面. ...
- Activiti工作流引擎开发系列
Activiti工作流引擎开发系列-01 作者:Jesai 没有伞的孩子,只能光脚奔跑! 前言: 初次接触工作流这个概念是自从2014年11月份开始,当时是由于我的毕业设计需要,还记得当时我毕业设计的 ...
- matplotlib 折线图
1.基本要点 # 导入模块 from matplotlib import pyplot as plt # x轴数据 x = range(2, 26, 2) # y轴数据 y = [15, 13, 14 ...
- JS中字符串切片
1.charAt 作用:根据索引值获取字符串 s1= "Hello world"; // 根据索引求字符 var myChar = s1.charAt(4); console.lo ...
- Linux下安装JDK 1.8
前言 JDK是 JAVA 的软件开发工具包,如果要使用JAVA来进行开发,或者部署基于其开发的应用,那么就需要安装JDK.本次将在Linux下安装JDK及配置环境. 本人环境:CentOS 7.3 6 ...
- chrome最耐看的主题
google chrome最耐看的主题James White大家可以尝试一下
- 「 扫盲 」Web服务器和应用服务器的区别
我们经常使用apache,tomcat,nginx,jetty等服务器,但并不清楚它们间的区别,它们中,哪些是Web服务器,哪些是应用服务器?今天就来告诉你 Web服务器 理解WEB服务器,首先你要理 ...
- Python通过win32 com接口实现offic自动化
最近几天通过Python做一些自动生成office报表的东东,比如解析.xml文件,导出.html/WORD/PPT等格式,html不足一提,只需要简单的html静态网页知识即可,这儿要说的是怎么生成 ...
- 分享一下我在mysql5.6+mysql8数据库安装过程中的一些坑!
Mysql5.6安装 下载好安装包后,在bin目录下用cmd打开,输入mysqld install [服务名]新建个服务 在windows+r输入services.msc即可查看服务 怎样使用mysq ...
- Vue使用better-scroll左右菜单联动
说明 最近想做一个vue商城小项目,练习一下vue的语法,刚刚好碰到了需要左右菜单实现联动,因此就接触了 better-scroll. github地址 中文文档. 代码 页面结构以及数据 //页面结 ...