Swift中的枚舉比OC中的枚舉更強(qiáng)大晨缴。
一. 枚舉的基本用法
二. 關(guān)聯(lián)值(Associated Values)
有時(shí)候?qū)⒚杜e的成員值跟其他類型的值關(guān)聯(lián)存儲(chǔ)在一起漆改,會(huì)非常有用互婿。
關(guān)聯(lián)值舉例:
三. 原始值(Raw Values)
枚舉成員可以使用相同類型的默認(rèn)值預(yù)先對(duì)應(yīng)警儒,這個(gè)默認(rèn)值叫做:原始值
上面的Character和String并不是什么繼承嘶炭,而是代表枚舉的原始值是什么類型挤忙。
注意:原始值不占用枚舉變量的內(nèi)存
- 隱式原始值(Implicitly Assigned Raw Values)
如果枚舉的原始值類型是Int、String,Swift會(huì)自動(dòng)分配原始值
如果原始值類型是String粹舵,那么原始值就是枚舉成員的名字钮孵,比如下面兩種寫法等價(jià):
如果原始值類型是Int,那么原始值從0開始齐婴,并且遞增油猫,如下:
枚舉可以根據(jù)rawValue來給枚舉賦值稠茂,但是沒有什么所謂的初始化器柠偶,比如:
let season = Season(rawValue: 1) //夏天
四. 遞歸枚舉(Recursive Enumeration)
一個(gè)枚舉成員里面又用到了這個(gè)枚舉,那么這種枚舉就叫遞歸枚舉睬关,遞歸枚舉必須要加上indirect關(guān)鍵字诱担,否則編譯器會(huì)報(bào)錯(cuò)。(indirect adj. 間接的电爹;迂回的)
下面是一個(gè)做算數(shù)運(yùn)算的遞歸枚舉蔫仙,枚舉成員分別是數(shù)字、加法丐箩、減法:
計(jì)算的是:(5 + 4) - 2 = 7摇邦。
五. MemoryLayout
可以使用MemoryLayout獲取數(shù)據(jù)類型占用的內(nèi)存大小
var pwd = Password.numer(9, 8, 6, 4)
pwd = .other
MemoryLayout.size(ofValue: pwd) // 至少需要33字節(jié)
MemoryLayout.stride(ofValue: pwd) // 實(shí)際系統(tǒng)分配40字節(jié)
MemoryLayout.alignment(ofValue: pwd) // 內(nèi)存對(duì)齊8
上面的枚舉,int占用8字節(jié)屎勘,case other只需1字節(jié)施籍,4 x 8 + 1 = 33字節(jié),又因?yàn)閷?duì)齊參數(shù)是8概漱,所以枚舉變量至少需要33字節(jié)丑慎,實(shí)際系統(tǒng)分配的是40字節(jié)。
可能你還有疑問瓤摧,既然case number(Int, Int, Int, Int)占用32字節(jié)竿裂,case other占用1字節(jié),那存儲(chǔ)other的時(shí)候直接用那32個(gè)字節(jié)不就好了嗎照弥?如果真的是other用那32個(gè)字節(jié)中的一個(gè)字節(jié)腻异,那么你就沒法分辨到底是other還是number了,所以必須要有一個(gè)字節(jié)用來分辨是other還是number这揣,所以必須是32 + 1 = 33字節(jié)捂掰。
- 思考下面枚舉變量的內(nèi)存布局
enum Season {
// 就相當(dāng)于存 0 1 2 3 就能分辨出是哪個(gè)枚舉值了
case spring, summer, autumn, winter
}
var s = Season.spring //相當(dāng)于存 0
var s1 = Season.summer //相當(dāng)于存 1
var s2 = Season.autumn //相當(dāng)于存 2
MemoryLayout.size(ofValue: s) // 1字節(jié)
MemoryLayout.stride(ofValue: s1) // 1字節(jié)
MemoryLayout.alignment(ofValue: s2) // 1字節(jié)
上面枚舉變量占用一個(gè)字節(jié),這也很容易理解曾沈,對(duì)于Season枚舉这嚣,最簡單的方法就是在內(nèi)存中存0 1 2 3就能表明是哪個(gè)枚舉成員了,所以只需要一個(gè)字節(jié)就可以塞俱。這也解釋了上面為什么說:原始值不占用枚舉變量的內(nèi)存姐帚。
如果加個(gè)Int原始值呢?
enum Season : Int {
// 就相當(dāng)于存 0 1 2 3 就能分辨出是哪個(gè)枚舉值了
case spring = 6, summer, autumn, winter
}
var s = Season.spring //相當(dāng)于存 0
var s1 = Season.summer //相當(dāng)于存 1
var s2 = Season.autumn //相當(dāng)于存 2
MemoryLayout.size(ofValue: s) // 1字節(jié)
MemoryLayout.stride(ofValue: s1) // 1字節(jié)
MemoryLayout.alignment(ofValue: s2) // 1字節(jié)
由于原始值不占用枚舉變量的內(nèi)存障涯,所以上面枚舉變量還是占用一字節(jié)罐旗。
關(guān)聯(lián)值和原始值的區(qū)別:
① 關(guān)聯(lián)值是和枚舉變量關(guān)聯(lián)起來的膳汪,占用枚舉變量的內(nèi)存,原始值不占用枚舉變量的內(nèi)存九秀。
② 關(guān)聯(lián)值是不固定的遗嗽,要根據(jù)外面?zhèn)鬟M(jìn)來的值決定,原始值是固定死的鼓蜒,既然是固定死的痹换,那么就沒必要存儲(chǔ)在枚舉變量內(nèi)存里面。
可能你會(huì)想都弹,既然原始值不占用枚舉變量的內(nèi)存娇豫,那么原始值存儲(chǔ)在哪里呢?其實(shí)完全沒必要糾結(jié)原始值存儲(chǔ)在哪里畅厢,因?yàn)樵贾低耆梢圆淮媛锓肓。厦婷杜e的rawValue偽代碼,如下:
func rawValue() -> Int {
if self == 6 return 6
if self == 7 return 7
if self == 8 return 8
if self == 9 return 9
}