Swift 3.0基础学习之枚举类型

发布时间 - 2026-01-11 00:08:23    点击率:

枚举语法

使用关键字 enum 定义一个枚举

enum SomeEnumeration {
 // enumeration definition goes here
}

例如,指南针有四个方向:

enum CompassPoint {
 case north
 case south
 case east
 case west
}

这里跟 c 和 objective-c 不一样的是,Swift 的枚举成员在创建的时候没有给予默认的整型值。所以上面代码中的东南西北并不是0到3,相反,不同的枚举类型本身就是完全成熟的值,具有明确定义的CompassPoint类型。

也可以声明在同一行中:

enum Planet {
 case mercury, venus, earth, mars, jupiter, saturn, uranus, neptune
}

枚举赋值:

var directionToHead = CompassPoint.west

一旦 directionToHead 明确为 CompassPoint 类型的变量,后面就可以使用点语法赋值:

directionToHead = .east

Switch 表达式的枚举值匹配

switch 表达式如下:

directionToHead = .south
switch directionToHead {
case .north:
 print("Lots of planets have a north")
case .south:
 print("Watch out for penguins")
case .east:
 print("Where the sun rises")
case .west:
 print("Where the skies are blue")
}
// Prints "Watch out for penguins"

当然这里也可以加上 default 以满足所有的情况:

let somePlanet = Planet.earth
switch somePlanet {
case .earth:
 print("Mostly harmless")
default:
 print("Not a safe place for humans")
}
// Prints "Mostly harmless"

关联值

在 Swift 中,使用枚举来定义一个产品条形码:

enum Barcode {
 case upc(Int, Int, Int, Int)
 case qrCode(String)
}

可以这样理解上面这段代码:定义一个叫 Barcode 的枚举类型,带有值类型(Int,Int,Int,Int)的 upc和值类型(String)

现在可以这样创建其中一种类型的条形码:

var productBarcode = Barcode.upc(8, 85909, 51226, 3)

同一产品的另外一个类型的条形码可以这样赋值:

productBarcode = .qrCode("ABCDEFGHIJKLMNOP")

可以用 switch 来查看两种不同的条形码类型:

switch productBarcode {
case .upc(let numberSystem, let manufacturer, let product, let check):
 print("UPC: \(numberSystem), \(manufacturer), \(product), \(check).")
case .qrCode(let productCode):
 print("QR code: \(productCode).")
}
// Prints "QR code: ABCDEFGHIJKLMNOP."

上面的写法也可以改为:

switch productBarcode {
case let .upc(numberSystem, manufacturer, product, check):
 print("UPC : \(numberSystem), \(manufacturer), \(product), \(check).")
case let .qrCode(productCode):
 print("QR code: \(productCode).")
}
// Prints "QR code: ABCDEFGHIJKLMNOP."

原始值

这里是一个保存原始 ASCII 值的枚举类型:

enum ASCIIControlCharacter: Character {
 case tab = "\t"
 case lineFeed = "\n"
 case carriageReturn = "\r"
}

和上面关联值类似,在枚举中也可以指定每个 case 的默认值(raw values)。

值得注意的是,原始值和关联值不一样,原始值是在第一次定义枚举代码的时候已经设置好的,所有同类型的枚举 case 的原始值都是一样的。而关联值是你创建一个基于枚举 case新的常量或者变量的时候设置的,可以在每次你创建的时候都使用不一样的值。

隐式分配的原始值

如果枚举中 case 的原始值是整型或者字符串的时候,你不需要给每个 case 分配原始值,Swift 会自动帮你分配好值。

例如:

enum Planet: Int {
 case mercury = 1, venus, earth, mars, jupiter, saturn, uranus, neptune
}

Planet.mercury 原始值为1,Planet.venus 拥有一个隐式的原始值为2,以此类推。

如果枚举的原始值是 string 类型,那么他的原始值就是 case 名称的文本,例如:

enum CompassPoint: String {
 case north, south, east, west
}
CompassPoint.south 的隐式原始值是"south", 以此类推。
let earthsOrder = Planet.earth.rawValue
// earthsOrder is 3
 
let sunsetDirection = CompassPoint.west.rawValue
// sunsetDirection is "west"

从原始值初始化

如果你定义一个原始值类型的枚举,这时枚举会自动创建一个带有原始值类型的初始化器(参数名称为 rawValue),例如:

let possiblePlanet = Planet(rawValue: 7)
// possiblePlanet is of type Planet? and equals Planet.uranus

不是所有的 Int 值都可以找到对应的 planet,所以原始值初始化器会返回一个 optional 的枚举 case,上面的例子中的 possiblePlanet 是 Planet? 类型。

如果你想找原始值为11的 planet,初始化器将返回 nil:

let positionToFind = 11
if let somePlanet = Planet(rawValue: positionToFind) {
 switch somePlanet {
 case .earth:
  print("Mostly harmless")
 default:
  print("Not a safe place for humans")
 }
} else {
 print("There isn't a planet at position \(positionToFind)")
}
// Prints "There isn't a planet at position 11"

递归枚举

递归枚举是一个包含有一个或多个枚举 case 的关联值枚举实例的枚举,使用关键字 indirect 标明某个枚举 case 是递归的。

例如,下面是一个保存简单算法表达式的枚举:

enum ArithmeticExpression {
 case number(Int)
 indirect case addition(ArithmeticExpression, ArithmeticExpression)
 indirect case multiplication(ArithmeticExpression, ArithmeticExpression)
}

你也可以直接把 indirect 写在枚举定义的最前面:

indirect enum ArithmeticExpression {
 case number(Int)
 case addition(ArithmeticExpression, ArithmeticExpression)
 case multiplication(ArithmeticExpression, ArithmeticExpression)
}

下面的代码是示例如何创建这个递归枚举:

let five = ArithmeticExpression.number(5)
let four = ArithmeticExpression.number(4)
let sum = ArithmeticExpression.addition(five, four)
let product = ArithmeticExpression.multiplication(sum, ArithmeticExpression.number(2))

应用到计算函数中:

func evaluate(_ expression: ArithmeticExpression) -> Int {
 switch expression {
 case let .number(value):
  return value
 case let .addition(left, right):
  return evaluate(left) + evaluate(right)
 case let .multiplication(left, right):
  return evaluate(left) * evaluate(right)
 }
}
 
print(evaluate(product))
// Prints "18"

上面例子的算法表达式是:(5 + 4) * 2,结果为18

参考英语原文:
https://developer.apple.com/library/content/documentation/Swift/Conceptual/Swift_Programming_Language/Enumerations.html#//apple_ref/doc/uid/TP40014097-CH12-ID145

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者使用Swift能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。


# swift3.0  # 枚举  # swift枚举类型  # 枚举类型  # swift中c风格的for循环执行效率  # 详解Swift编程中的for循环的编写方法  # Swift学习教程之SQLite的基础使用  # RxSwift学习教程之基础篇  # Swift 3.0基础学习之扩展  # Swift之for循环的基础使用学习  # 递归  # 是一个  # 的是  # 如果你  # 值为  # 以此类推  # 隐式  # 创建一个  # 整型  # 是在  # 多个  # 你也  # 两种  # 可以用  # 帮你  # 这段  # 可以直接  # 英语  # 另外一个  # 这篇文章 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel怎么配置S3云存储驱动_Laravel集成阿里云OSS或AWS S3存储桶【教程】  Laravel如何使用Livewire构建动态组件?(入门代码)  猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?  微信小程序 五星评分(包括半颗星评分)实例代码  微信小程序 scroll-view组件实现列表页实例代码  微信小程序 input输入框控件详解及实例(多种示例)  Laravel如何实现全文搜索_Laravel Scout集成Algolia或Meilisearch教程  logo在线制作免费网站在线制作好吗,DW网页制作时,如何在网页标题前加上logo?  Laravel怎么实现微信登录_Laravel Socialite第三方登录集成  中国移动官方网站首页入口 中国移动官网网页登录  深圳防火门网站制作公司,深圳中天明防火门怎么编码?  php json中文编码为null的解决办法  消息称 OpenAI 正研发的神秘硬件设备或为智能笔,富士康代工  Java解压缩zip - 解压缩多个文件或文件夹实例  品牌网站制作公司有哪些,买正品品牌一般去哪个网站买?  音响网站制作视频教程,隆霸音响官方网站?  Laravel怎么生成二维码图片_Laravel集成Simple-QrCode扩展包与参数设置【实战】  谷歌Google入口永久地址_Google搜索引擎官网首页永久入口  高配服务器限时抢购:企业级配置与回收服务一站式优惠方案  如何用已有域名快速搭建网站?  香港服务器部署网站为何提示未备案?  如何快速搭建支持数据库操作的智能建站平台?  Win11搜索不到蓝牙耳机怎么办 Win11蓝牙驱动更新修复【详解】  Laravel如何升级到最新的版本_Laravel版本升级流程与兼容性处理  Laravel软删除怎么实现_Laravel Eloquent SoftDeletes功能使用教程  如何基于PHP生成高效IDC网络公司建站源码?  Laravel如何使用Scope本地作用域_Laravel模型常用查询逻辑封装技巧【手册】  如何生成腾讯云建站专用兑换码?  Laravel如何处理JSON字段的查询和更新_Laravel JSON列操作与查询技巧  Win11怎么关闭透明效果_Windows11辅助功能视觉效果设置  详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)  JavaScript如何实现类型判断_typeof和instanceof有什么区别  b2c电商网站制作流程,b2c水平综合的电商平台?  Laravel如何实现API速率限制?(Rate Limiting教程)  nodejs redis 发布订阅机制封装实现方法及实例代码  Android Socket接口实现即时通讯实例代码  如何用VPS主机快速搭建个人网站?  Laravel Sail是什么_基于Docker的Laravel本地开发环境Sail入门  Laravel路由Route怎么设置_Laravel基础路由定义与参数传递规则【详解】  详解Huffman编码算法之Java实现  Laravel的契約(Contracts)是什么_深入理解Laravel Contracts与依赖倒置  Laravel如何使用集合(Collections)进行数据处理_Laravel Collection常用方法与技巧  如何挑选高效建站主机与优质域名?  Laravel如何安装使用Debugbar工具栏_Laravel性能调试与SQL监控插件【步骤】  手机软键盘弹出时影响布局的解决方法  javascript读取文本节点方法小结  Win11关机界面怎么改_Win11自定义关机画面设置【工具】  长沙做网站要多少钱,长沙国安网络怎么样?  如何注册花生壳免费域名并搭建个人网站?  Laravel怎么集成Vue.js_Laravel Mix配置Vue开发环境