文章的缘由
从C、C++、OC、或是Java转过来跟接触Swift中枚举的时候,我想都会有这么一个疑问,一个在其他语言中简简单单的枚举,为什么在Swift中复杂化了,这种复杂化背后的意义如何呢?面对Enum中新添加的特性,如何更好的使用?这些都是我在学习的过程中的思考,或许还有更多的部分需要探索,期待与您的交流
我理解的Swift中的Enum
一般关于枚举的定义是:【0】定义一组相关的常量值(c,c++,oc中这个定义的含义特别明显)
而Swift中把这个定义扩展到更广泛层次,我的理解是:只能定义常量实例的类型。有点拗口,我解释一下
【1】你定义了一个enum,它是一个枚举类型:“它的实例所封装的内容不能被修改。”
关于关联值
c,c++,oc中的枚举是没有关联值的,Java中的枚举可以有相关的属性,相对于这些语言,可见Swift中的关联值的功能更加开放。那么提供关联值的意义何在?我的理解是使Swift中的枚举更加类化,对于枚举的抽象层次加深,它提供给我们一个生成常量实例的类。
举个例子:
<pre>
enum StarbucksCoffee{
case Short(CGFloat)
case Tall(CGFloat)
case Grande(CGFloat)
case Venti(CGFloat)
}
</pre>
星巴克的咖啡,每次卖出一杯,不管是小,中,大,还是超大杯,在卖出这一刻它所含的咖啡的毫升量是不可能再变化的。
<pre>
let shortCoffee = StarbucksCoffee.Short(236)
let ventiCoffee = StarbucksCoffee. Venti(591)
</pre>
即使我把let换成var,都不用担心shortCoffee/ventiConffee所引用实例的咖啡含量会被改变。使用其他语言实现至少需要const或者final进行约束,增加了逻辑的复杂度。
关于模式匹配
注意看这里不是值的匹配,而是模式的匹配。这也是与其他语言不同的地方,其中语言中的Switch语句都是进行值的相等性匹配。
回头再来看定义【0】我们就发现Swift中enum的定义应该被修改:
【2】定义了一组生成常量的模式
无论是从上面StarbucksCoffee的例子,还是
<pre>
enum Trade {
case Buy(stock: String, amount: Int)
case Sell(stock: String, amount: Int)
}
</pre>
例子中都可以看出枚举的每一个enum element 不再是具体的值,而是一个个case(模式)
即使是最原始的枚举
<pre>
enum Movement{
case Left
case Right
case Top
case Bottom
}</pre>
显然也满足这个定义。我对这种匹配的理解不是值的匹配,也不是模式的匹配,而是程序实现方法和思路的匹配。
枚举中的方法和静态方法
我更愿意理解为helper method,这里方法可以使你处理问题,更加的巧妙优美。
举两个例子:
关于普通方法:处理一些静态数据(不变的数据)。一个例子来源于tableView的静态数据:例子来源于:https://www.natashatherobot.com/swift-enums-tableviews/
<pre>
enum MinionIndex: Int {
case DAVE, BOB, JERRY, JORGE, KEVIN, MARK, PHIL, STUART, TIM
static let minionNames = [
DAVE : "Dave", BOB : "Bob", JERRY : "Jerry",
JORGE : "Jorge", KEVIN : "Kevin", MARK : "Mark",
PHIL : "Phil", STUART : "Stuart", TIM : "Tim"]
func minionName() -> String {
if let minionName = MinionIndex.minionNames[self] {
return minionName
} else {
return "Minion"
}
}
func minionImage() -> UIImage {
return UIImage(named: "Minion(minionName())")
}
}
override func tableView(tableView: UITableView!, cellForRowAtIndexPath indexPath: NSIndexPath!) -> UITableViewCell! {
let cell = tableView.dequeueReusableCellWithIdentifier(minionCellIdentifier) as UITableViewCell
if let minionIndex = MinionIndex.fromRaw(indexPath.row) {
cell.textLabel.text = minionIndex.minionName()
cell.imageView.image = minionIndex.minionImage()
}
return cell
}
</pre>
关于静态方法:
可以作为静态工厂方法,根据传进来的值,生成特定的枚举实例
<pre>
enum Device {
case AppleWatch
static func fromSlang(term: String) -> Device? {
if term == "iWatch" {
return .AppleWatch
}
return nil
}
}
print (Device.fromSlang("iWatch"))
</pre>