★ iOS高级:Swift入门精讲③ 01 swift编程-06-枚举-05答疑

前言:

本篇仅为视频学习笔记

短接上篇:
★ iOS高级:Swift入门精讲③ 01 swift编程-05-枚举-04原始值内存分布

★ 这个原始值,它存储到什么地方呢?(无需纠结)
enum Season: Int {
case spring = 1, summer, autumn, winter
}

 var s = Season.spring
 var s1 = Season.spring
 var s2 = Season.spring
 
 s2.rawValue
 
 MemoryLayout.size        // 1,实际用到的空间大小
 MemoryLayout.stride      // 1,分配占用的空间大小
 MemoryLayout.alignment   // 1,对齐参数

我们把它改成上面代码那样,冒号Int。(enum Season: Int)。不就是代表枚举Season的原始值是Int类型吗,所以呢,我们写了一个原始值spring = 1。summer, autumn, winter依次默认为2、3、4。

之前,说过,这个原始值不是存储到s变量里面去的。像这些枚举变量。它只占用一个字节,它一个字节就能搞的清楚,它是spring,是summer,是autumn,还是winter。

那你可能就会想,那么这个原始值存储到哪里呢?其实这个东西,你思考一下,它不存也可以啊,怎么不存也可以呢?

因为,你想想,你拿原始值不过就是调用 s2.rawValue。那么这个rawValue,完全怎么可以实现呢?

完全可以这样实现,如下代码:

 enum Season: Int {
     case spring = 1, summer, autumn, winter
     
     func rawValue() -> Int {
         if self == 0 return 1
         if self == 1 return 2
         if self == 2 return 3
         if self == 3 return 4
     }
 }

所以,完全可以不用思考,它存在什么地方,它完全可以不用存。

是不是可以这个样子,如果发现我自己内存里面存储的是0,也就是存储的是spring,我们就return 1。那么,如果我发现自己内存里,存储的是1,不就意味着我是summer,我是summer,那么我的原始值就是return 2。那么我这个rawValue获取原始值的实现内部完全可以如上面代码那样写。


★ iOS高级:Swift入门精讲③ 01 swift编程-06-枚举-05答疑_第1张图片

所以,大家没有必要去纠结,如果这里写的是enum Season:String,那么你关联的这个原始值,它存储到什么地方呢?它不一定需要存储。所以,这个放哪里,这个跟你无关。你只要搞清楚,这个枚举变量占用多少内存就可以了。

★ Password枚举为什么实际用到的空间大小为33个字节呢?

 enum Password {
     case number(Int,Int,Int,Int)  // 32
     case other  // 1
 }
 
 var pwd  = Password.number(5, 6, 4, 7)  // 占用32字节
   pwd = .other
 var pwd1 = Password.number(2, 4, 3, 0)  // 占用32字节
 var pwd2 = Password.number(23, 43, 133,30)  // 占用32字节

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

再说另外一个问题,刚刚说到Password这个枚举变量的话,那么我定义一个pwd枚举变量。最后真正分配的内存是多少? 是40,但实际上真正利用起来的是33。其实这个33是怎么利用起来的呢?

我们在 case number(Int,Int,Int,Int) 这个位置标记一个32,是没有问题的。但是我们在case other // 1标记一个1的话,其实不是很严谨。那么这个1,这个字节是怎么用的呢?

其实,有的人会怎么想呢?它会想,你这个pwd这个枚举变量,只占用32个字节,不就够了吗?为什么呢?首先,如果你的pwd是 case number(Int,Int,Int,Int) 这种类型的话4个证书,刚好是32个字节,存储到里面去。但是,如果你是other的话,大家有可能会想,像下面这种:

 enum Password {
     case other, abc, ddd
 }

非常普通的这些成员,他是不是只需要一个字节就够了。比如说,other是0、abc是1,ddd是2。是不是用一个字节,就能把这些数值表达清楚了。那么,一说到这个,大家可能会想,那么你存储other只需要1个字节,但是在var pwd = Password.number(5, 6, 4, 7) 这行时,我们不是已经分配了32个字节给pwd了吗?

那么你的.other赋值给pwd。( pwd = .other ),就不能将这个other,1个字节的数据存储到var pwd = Password.number(5, 6, 4, 7) 这32个字节里面去吗?肯定会有这样的问题,有这样的问题说明你还没想透。

你思考一下,如果你将 .other 这一个值。假设你这个值 case other是 0。那么,你能把这个0存储到刚刚 Password.number(5, 6, 4, 7) 32个字节里面去吗?
那我就问你将来你怎么判断呢?

★ iOS高级:Swift入门精讲③ 01 swift编程-06-枚举-05答疑_第2张图片

你怎么知道,它是.otner还是 .number呢?根本就分不清。请问我现在给你32个字节,你怎么知道32个字节里面存储的是.otner这个家伙还是这(5, 6, 4, 7) 四个家伙。

因为你拿到32,完全可以把这32个字节分成4个部分,每个部分8个字节,然后分别认为是这(Int,Int,Int,Int) 四个整型。

我也可以拿到32个字节中,最前面的一个字节,当作是 .other。所以,你必须var pwd = Password.number(5, 6, 4, 7) // 32 + 1 其中的一个字节是来告诉你是case other,这个类型,还是case number(Int,Int,Int,Int) // 32 这个类型。你可以这样思考,这是用来辨别的。就是从字节中,我们可以看得出来,我们整体数据是case number(Int,Int,Int,Int) 这个家伙,还是case other这个家伙。

所以,只给它enum Password 32个字节是不行的。所以,必然是32 + 1,那么,你可能会想为什么不是 32 + 2呢?没必要这么浪费,你搞出一个来,就行了。

既然,你需要占用33个字节,再加上它的内存对齐参数是8个字节,所以最后分配给它的是40个字节。

你可能感兴趣的:(★ iOS高级:Swift入门精讲③ 01 swift编程-06-枚举-05答疑)