深入探究Swift枚举关联值的内存

enum Season {
 case Spring, Summer, Autumn, Winter
}
let s = Season.Spring

这是枚举最基础的用法,但是在swift中,对枚举的功能进行了加强,也就是关联值。

关联值可以将额外信息附加到 enum case中,像下面这样子。

enum Test {
 case test1(v1: Int, v2: Int, v3: Int)
 case test2(v1: Int, v2: Int)
 case test3(v1: Int)
 case test4
}
let t = Test.test1(v1: 1, v2: 2, v3: 3)

switch t {
case .test1(let v1, let v2, let v3):
 print(v1, v2, v3)
default:
 break
}
// 输出: 1 2 3

我们可以看到,在我们创建一个枚举值t的时候,设置他的选项为test1,同时可以关联3个Int类型的值,然后在switch中,我们还可以把这3个Int值取出来进行使用。

我们今天的主要任务就是探索一下有关联值的枚举类型,再底层的内存布局是什么样子的,这些值都是怎么储存的。

在OC中我们使用sizeOf此类方法,可以输出一个变量占用内存的大小,在swift中也有此类的工作类,那就是MemoryLayout。

print(MemoryLayout<Int>.size)// 实际使用内存大小
print(MemoryLayout<Int>.stride)//分配内存大小
print(MemoryLayout<Int>.alignment)//内存对其参数

// 输出 8 8 8 

上面的例子是只是简单的实例MemoryLayout的用法,这个我们知道,在64位的系统中Int类型确实是占用8个字节(64位)。接下来我们就看一下枚举的内存占用情况。

点击Xcode菜单栏中的Debug -> Debug Workflow -> View Memory,然后在下面红色框中输入变量的内存地址,就可以看到变量的内存使用情况。

使用swift后,从xcode没法直接打印变量的内存地址,这里我们使用了github上的一个工具类来帮助我们输出变量的内存地址。

准备工作完成后,我们先从最基础的枚举开始。

enum Season {
 case Spring, Summer, Autumn, Winter
}
print("实际占用:",MemoryLayout<Season>.size)
print("分配:",MemoryLayout<Season>.stride)
print("对齐参数:", MemoryLayout<Season>.alignment)

var s = Season.Spring
print("内存地址",Mems.ptr(ofVal: &s))

print("内存数据",Mems.memStr(ofVal: &s, alignment: .one))

s = Season.Summer
print("内存数据",Mems.memStr(ofVal: &s, alignment: .one))

s = Season.Autumn
print("内存数据",Mems.memStr(ofVal: &s, alignment: .one))

s = Season.Winter
print("内存数据",Mems.memStr(ofVal: &s, alignment: .one))

注:Mems.memStr可以直接打印内存数据,这样我们就不用每次拿到地址再去工具中看了

实际占用: 1
分配: 1
对齐参数: 1
内存地址 0x00007ffee753f0f0
内存数据 0x00
内存数据 0x01
内存数据 0x02
内存数据 0x03

我们可以看到这种普通的枚举类型,只占用一个字节。而且通过我们对变量设置不同的枚举值,打印的这一个字节的数据也是不同的,其实也就是使用这一个字节通过设置不同的数值来表示不同的枚举值,这样的话其实可以至少储存0x00-0xFF共256个值。那如果超过256个case呢?其实我觉得没有必要考虑这种情况,枚举本来设计出就是为了区分有限中情况,如果太多,就像200多个,那完全可以使用Int来设置不同的值了,就没必要用枚举了,当然,如果您愿意探究一下的话也是可以的。

接下来我们使用一个带关联值的枚举来看一下。

enum Test {
 case test1(v1: Int, v2: Int, v3: Int)
 case test2(v1: Int, v2: Int)
 case test3(v1: Int)
 case test4
}

print("实际占用:",MemoryLayout<Test>.size)
print("分配:",MemoryLayout<Test>.stride)
print("对齐参数:", MemoryLayout<Test>.alignment)

var t = Test.test1(v1: 1, v2: 2, v3: 3)
print("内存地址",Mems.ptr(ofVal: &t))

print("内存数据",Mems.memStr(ofVal: &t, alignment: .one))

t = Test.test2(v1: 4, v2: 5)
print("内存数据",Mems.memStr(ofVal: &t, alignment: .one))

t = Test.test3(v1: 6)
print("内存数据",Mems.memStr(ofVal: &t, alignment: .one))

t = Test.test4
print("内存数据",Mems.memStr(ofVal: &t, alignment: .one))

下面是输出, 为了能直观一下,我给插了几个换行

实际占用: 25
分配: 32
对齐参数: 8
内存地址 0x00007ffee0afe0d8
内存数据
0x01 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x02 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x03 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x00
0x00 0x00 0x00 0x00 0x00 0x00 0x00
内存数据
0x04 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x05 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x01
0x00 0x00 0x00 0x00 0x00 0x00 0x00
内存数据
0x06 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x02
0x00 0x00 0x00 0x00 0x00 0x00 0x00
内存数据
0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00
0x03
0x00 0x00 0x00 0x00 0x00 0x00 0x00

实际占用了25个字节,我们至少可以确定,枚举的关联值是存储在枚举值的内存中的。

但是通过这一个例子其实可能还看不出有什么规律,大家可以多用几个例子来验证,这是我就直接说结论了。

有关联值得枚举实际占用的内存是最多关联值占用的内存+1,在我们这个Test中,test1的关联值是最多的,有3个Int类型的关联值,所以要8*3=24字节来存放关联值,但是还需要一个字节来储存(辨别)是哪一个case。

带着这个结论我们看一下输出的结果:

当t=.test1时,前面24个字节分配给3个Int类型关联值,分别存储了1,2,3, 第25个字节是0。

当t=.test2时,前面24个字节还是留给关联值的,但是test2只有两个关联值,所以使用了前面16个字节分配给他的关联值,此时17到24这8字节就空置,第25个字节是1。

...

最后当t = test4 , 没有关联值,所以前面的字节都是0, 只有第25个字节是3

以此类推...

第25个字节其实完全可以看成一个辨识位,或者说第25个字节就是枚举的本质,通过不同值来区分不同case,只是因为有了关联值,所以开辟了更多的空间来存储而已。

后面多余的字节都是为了内存对齐,内存对其相关的知识大家可以自行上网查阅。

补充:

既然说到了关联值,那就顺便对枚举原始值说两句。具通过你打印带原始值的枚举的内存数据,发现是否带有原始值对枚举的内存占用并无影响,所以原始值应该不是存储在枚举变量的内部的。大家可以自己试验一下

总结

到此这篇关于深入理解Swift枚举关联值内存的文章就介绍到这了,更多相关Swift枚举关联值的内存内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Swift教程之枚举类型详解

    枚举定义了一个常用的具有相关性的一组数据,并在你的代码中以一个安全的方式使用它们. 如果你熟悉C语言,你就会知道,C语言中的枚举指定相关名称为一组整数值.在Swift中枚举更为灵活,不必为枚举的每个成员提供一个值.如果一个值(被称为"原始"的值)被提供给每个枚举成员,则该值可以是一个字符串,一个字符,或者任何整数或浮点类型的值. 另外,枚举成员可以指定任何类型,每个成员都可以存储的不同的相关值,就像其他语言中使用集合或变体.你还可以定义一组通用的相关成员为一个枚举,每一种都有不同的一组

  • Swift 3.0基础学习之枚举类型

    枚举语法 使用关键字 enum 定义一个枚举 enum SomeEnumeration { // enumeration definition goes here } 例如,指南针有四个方向: enum CompassPoint { case north case south case east case west } 这里跟 c 和 objective-c 不一样的是,Swift 的枚举成员在创建的时候没有给予默认的整型值.所以上面代码中的东南西北并不是0到3,相反,不同的枚举类型本身就是完全

  • 深入解析Swift编程中枚举类型的相关使用

    枚举是由用户定义的数据类型的一组相关值.关键字 enum 用来定义枚举数据类型. 枚举功能 枚举在 swift 也类似于 C 和 Objective C 中结构类型 它是在一个类中声明,其值是通过该类的实例来访问 初始成员值是用枚举初始化定义的 其功能也扩展确保标准的协议功能 语法 枚举引入 enum 关键字和一对大括号内将它们定义: 复制代码 代码如下: enum enumname {    // enumeration values are described here } 例如,可以为星期

  • Swift枚举的一些小用法总结

    前言 在 Swift 中,枚举是一个非常方便也非常强大的类型.我们在日常使用中也经常会使用到它. 例如,我们最常见的 optional: enum Optional<T> { case Some(T) case None } 这里不准备介绍枚举的基本用法,只是记录两个比较好用的枚举用法. 关联值 关联值是将额外信息附加到 enum case 中的一种极好的方式. 例如,当我们需要将一系列的值传到下一个类中时,一般情况下我们像下方代码一样写出几个设置的方法: struct MyStruct {

  • 详解Swift中enum枚举类型的用法

    一.引言 在Objective-C语言中,没有实际上是整型数据,Swift中的枚举则更加灵活,开发者可以不为其分配值类型把枚举作为独立的类型来使用,也可以为其分配值,可以是字符,字符串,整型或者浮点型数据. 二.枚举语法 Swift中enum关键字来进行枚举的创建,使用case来创建每一个枚举值,示例如下: //创建姓氏枚举,和Objective-C不同,Swift枚举不会默认分配值 enum Surname { case 张 case 王 case 李 case 赵 } //创建一个枚举类型的

  • Swift编程之枚举类型详解

    想必写过程序的童鞋对枚举类型并不陌生吧,使用枚举类型的好处是多多的,在这儿就不做过多的赘述了.Fundation框架和UIKit中的枚举更是数不胜数,枚举可以使你的代码更易阅读并且可以提高可维护性.在Swift语言中的枚举可谓是让人眼前一亮.在Swift中的枚举不仅保留了大部分编程语言中枚举的特性,同时还添加了一些好用而且实用的新特性,在本篇文章中将领略一些Swift中枚举类型的魅力. 有小伙伴会问,不就是枚举么,有什么好说的.在Swift中的枚举怎不然,Swift中的枚举的功能要强大的多,不仅

  • 深入探究Swift枚举关联值的内存

    enum Season { case Spring, Summer, Autumn, Winter } let s = Season.Spring 这是枚举最基础的用法,但是在swift中,对枚举的功能进行了加强,也就是关联值. 关联值可以将额外信息附加到 enum case中,像下面这样子. enum Test { case test1(v1: Int, v2: Int, v3: Int) case test2(v1: Int, v2: Int) case test3(v1: Int) cas

  • Swift在什么情况会发生内存访问冲突详解

    前言 众所周知,Swift 是一门类型安全的语言,它会通过编译器报错来阻止你代码中不安全的行为.比如变量必须在使用之前声明.变量被销毁之后内存不能在访问.数组越界等问题. Swift 会通过对于修改同一块内存,同一时间以互斥访问权限的方式(同一时间,只能有一个写权限),来确保你的代码不会发生内存访问冲突.虽然 Swift 是自动管理内存的,在大多数情况下你并不需要关心这个.但理解何种情况下会发生内存访问冲突也是十分必要的. 首先,来看一下什么是内存访问冲突. 内存访问冲突 当你设值或者读取变量的

  • 深入探究Java编程是值传递还是引用传递

    目录 1.基本数据类型的参数传递 2.引用数据类型的参数传递 3.原理 文章目的:验证Java语言到底是值传递还是引用传递以及Java参数传递的实现原理. 问题引入: 先阅读代码段: public static void main(String[] args){ Person p=new Person("张三"); f(p); System.out.println("实参:"+p); } public static void f(Person p){ p.name=

  • ASP.NET堆和栈一之基本概念和值类型内存分配

    ".NET的堆和栈"系列: ASP.NET堆和栈一之基本概念和值类型内存分配 ASP.NET堆和栈二之值类型和引用类型参数传递和内存分配 ASP.NET堆和栈三之引用类型对象拷贝和内存分配 ASP.NET堆和栈四之对托管和非托管资源垃圾回收和内存分配 当我们对.NET Framework的一些基本面了解之后,实际上,还是很有必要了解一些更底层的知识.比如.NET Framework是如何进行内存管理的,是如何垃圾回收的......这样,我们才能写出更高性能的程序. 在.NET Fram

  • Swift编程中用以管理内存的自动引用计数详解

    Swift 内存管理功能是通过使用自动引用计数(ARC)来处理.ARC用于初始化和取消初始化所述系统资源,从而释放使用的类实例的存储器空间当实例不再需要.ARC跟踪代码的实例有效地管理存储资源之间的关系的信息. ARC的功能 在每一次一个新的类实例被创建时ARC分配一块内存以存储信息 init() 关于实例类型和其值的信息存储在存储器中 当类实例不再需要它自动由 deinit() 释放,用于进一步类实例的存储和检索的存储空间 ARC保存在磁道当前参照类实例的属性,常量和变量,使得 deinit(

  • Swift里的值类型与引用类型区别和使用

    Swift里面的类型分为两种: ●值类型(Value Types):每个实例都保留了一分独有的数据拷贝,一般以结构体 (struct).枚举(enum) 或者元组(tuple)的形式出现. ●引用类型(Reference Type):每个实例共享同一份数据来源,一般以类(class)的形式出现. 在这篇博文里面,我们会介绍两种类型各自的优点,以及应该怎么选择使用. 值类型与引用类型的区别 值类型和引用类型最基本的分别在复制之后的结果.当一个值类型被复制的时候,相当于创造了一个完全独立的实例,这个

  • Swift中Optional值的链式调用学习笔记

    Swift中的Optional值有这样的特性,当对其进行可选拆包时,即使用?进行Optional类型值的取值时,如果Optional值不为nil,则会返回原始类型的数据值,如果为nil,则会返回nil.因此,当使用?对Optional拆包后进行方法.属性或者下标的调用时,如果有值,则会成功相应调用,如果没有值,则会调用失败,返回nil. 注意:使用!则会进行强制拆包,这时如果Optional值为nil,则会出现运行时错误,因此开发者在使用!进行强制拆包时,必须确认Optional类型值不为nil

  • swift中可选值?和!使用的方法示例

    Optional 可选值 Optional是 Swift 的一大特色,也是 Swift 初学者最容易困惑的问题. 定义变量时,如果指定该变量是可选的,表示该变量可以有一个指定类型的值,也可以是 nil. 此外,Swift的nil也和Objective-C有些不一样,在Objective-C中,只有对象才能为nil,而在Swift里,当基础类型(整形.浮点.布尔等)没有值时,也是nil,而不是一个初始值,没有初始值的值,是不能使用的,这就产生了Optional类型.定义一个Optional的值很容

随机推荐