[转自:http://www.th7.cn/Program/IOS/201411/319644.shtml]
成员值、原始值(符号所代表的内容)、哈希值:
1) 成员值和哈希值的关系:上述的Monday、Tuesday等都是成员值,但是和C语言等不一样,Swift的成员值不是整型的,Monday等并不代表任何类型,它仅仅是一个符号,因此这里Monday到Friday并不是整数0 ~ 4,为了安全(防止用户整型值和成员值乱用)Swift隐藏了这一实现,实际上Swift仍然使用整型值来管理成员值的,Swift使用的是哈希值,第一个成员值的哈希值为0,往后依次加1,可以使用Weekdays.Monday.hashValue来访问成员值所对应的哈希值,这个哈希值是不能改变的,由编译器自动生成,之后不可改变,Swift在背后实际上使用哈希值来识别枚举符号的(即成员),可以说类似于编译时作这样的处理:
#define Monday 0
#define Tuesday 1
#define Thursday 2
#define Friday 3谨记!成员值仅仅是一组抽象的符号,不能参与任何运算,也不代表任何数据类型!
2) 成员值仅仅是用来在程序中作为表示的符号,其目的仅仅是为了方便理解程序,但有时候需要给这些符号赋予实际意义,比如在上例中Monday仅仅是一个符号,但是打印的时候想用”一“来代替它,而这个”一“就是Monday所代笔的实际内容,如果直接打印.Monday编译器是不允许的,因此需要给它定义一个实际意义的值来代表它,而这个就是成员值的默认值了;
3) 由上述概念的描述可以总结出一旦在定义枚举时指定了原始值之后是再也都无法修改的,因为原始值就是枚举符号的实际内容,比如成员值juice的原始值设为”橙汁",在程序中用juice进行相关操作当然比较方便,但是输出的时候希望输出"橙汁"这个实际内容,因此作为juice实际内容的"橙汁"是无法改变,如果改变不仅违反逻辑也违反编译器的规则(会直接报错的),同时Swift在语法上也根本不支持改变成员的原始值;
要想定义原始值就必须得给枚举类型指定一种数据类型来让所有的成员的原始值的类型保持一致,Swift不允许各个成员的原始值的类型不同(因为枚举本身就是指一组意义相似的元素,如果成员的类型都不一样(有的是Int有的时String)不就违反了这个根本的逻辑原则了,因此Swift语法上要禁止这种情况的发生):
enum WeekDays: String{
case Monday = "星期一"
case Tuesday = "星期二"
case Wednesday = "星期三"
case Thursday = "星期四"
case Friday = "星期五"}
var day = WeekDays.Mondayprintln(day.rawValue + "好!") // 星期一好!在这种情况下每个成员值的原始值都必须定义!
4) 原始值的推断:在Swift中只有Int型的原始值可以推断,其余类型包括Double、String、Character类型都无法在原始值中推断;
这里的推断是指不用给出所有成员值的原始值而只需要给定一部分即可,其余的原始值Swift可以自动推断出,但是这里就只有Int类型的支持原始值推断,而推断的方法和C语言的枚举类型一样:
enum WeekDays: Int{
case Monday
case Tuesday
case Wednesday
case Thursday
case Friday}
print(WeekDays.Monday.rawValue)
print(WeekDays.Tuesday.rawValue)
print(WeekDays.Wednesday.rawValue)
print(WeekDays.Thursday.rawValue)
print(WeekDays.Friday.rawValue)// 不给出任何原始值,自动推断为0 1 2 3 4
enum Week: Int{
case Wednesday = 37
case Thursday
case Friday
case Saturday = 23
case Sunday}
print(Week.Monday.rawValue)
print(Week.Tuesday.rawValue)
(Week.Wednesday.rawValue)
(Week.Thursday.rawValue)
print(Week.Friday.rawValue)
print(Week.Saturday.rawValue)
print(Week.Sunday.rawValue)
// 0 1 37 38 39 23 24
// 第一个给出指定原始值之前的从0开始计// 所有给出原始值的,从给出的值开始往后加1// 遇到新的给出的原始值重新计算