适度给类型起别名能够让代码更加易懂,开发效率更高,可维护性更好。

一、typealias(别名)

typealias用来给类型起别名。

typealias Byte = Int8
typealias Short = Int16
typealias Long = Int64 typealias Date = (year: Int, month: Int, day: Int)
func test(_ date: Date) {
print(date.year)
}
test((2019, 6, 25))
// 输出:2019 typealias IntFn = (Int, Int) -> Int
func diff(v1: Int, v2: Int) -> Int {
v1 - v2
}
let fn: IntFn = diff
fn(10, 5)
// 输出:5

Void的本质就是空元祖的别名:public typealias Void = ()

二、枚举

Swift枚举和C/OC语言不一样,以前写OC的时候枚举本质是int类型,但Swift中枚举可以是多类型的。

官方建议:枚举名称使用大写,成员使用小写。

2.1. 基本用法

// 定义枚举
enum YBColor {
case white
case black
case gray
} // 等价于上面的代码
//enum YBColor {
// case white, black, gray
//} var color = YBColor.white
color = YBColor.black
color = .gray // 简写(因为此时已经确定变量color是YBColor类型)
print(color) // 输出:gray // 循环控制
switch color {
case .white:
print("white")
case .black:
print("black")
case .gray:
print("gray")
}

2.2. 关联值

有时将枚举的成员值其他类型的值关联存储在一起,会非常有用.

案例:

enum Score {
case points(Int)
case grade(Character)
} // 数值表达
var score = Score.points(96)
// 等级/字符表达
score = .grade("A") enum Date {
case digit(year: Int, month: Int, day: Int)
case string(String)
} var date = Date.digit(year: 2019, month: 06, day: 25)
date = .string("2019-06-25")
switch date {
case .digit(let year, let month, let day):
print(year, month, day, separator:"/")
case let .string(value):
print(value)
}
/*
输出:
2019-06-25
*/

let写在枚举成员前面意味着枚举成员形参只能是常量,放在形参里面可以自定义选择是var还是let

2.2. 原始值

枚举成员可以使用相同类型的默认值预先关联,这个默认值叫做:原始值。

enum Direction : Character {
case up = "w"
case down = "s"
case left = "a"
case right = "d"
}
var direction = Direction.up
print(direction) // 输出:up
print(direction.rawValue) // 输出:w
print(Direction.down.rawValue) // 输出:s

如果枚举的原始值类型是IntString,Swift会自动分配原始值:

enum Direction : String {
case up = "up"
case down = "down"
case left = "left"
case right = "right"
}
var direction = Direction.up
print(direction) // 输出:up
print(direction.rawValue) // 输出:up
print(Direction.down.rawValue) // 输出:down // 等价
enum Direction : String {
case up, down, left, right
}
var direction = Direction.up
print(direction) // 输出:up
print(direction.rawValue) // 输出:up
print(Direction.down.rawValue) // 输出:down

Int类型,成员值自增(类似C/OC枚举):

enum Season : Int {
case spring, summer, autumn, winter
}
print(Season.spring.rawValue) // 输出:0
print(Season.summer.rawValue) // 输出:1
print(Season.autumn.rawValue) // 输出:2
print(Season.winter.rawValue) // 输出:3 enum Season : Int {
case spring = 1, summer, autumn = 4, winter
}
print(Season.spring.rawValue)
print(Season.summer.rawValue)
print(Season.autumn.rawValue)
print(Season.winter.rawValue)

2.3. 递归枚举

  • 关键字:indirect
  • 可以把需要递归枚举的成员前面加indirect,也可以为了方便直接加到枚举定义前面。
indirect enum ArithExpr {
case number(Int)
case sum(ArithExpr, ArithExpr)
case diff(ArithExpr, ArithExpr)
} //enum ArithExpr {
// case number(Int)
// indirect case sum(ArithExpr, ArithExpr)
// indirect case diff(ArithExpr, ArithExpr)
//} let five = ArithExpr.number(5)
let four = ArithExpr.number(4)
let two = ArithExpr.number(2)
let sum = ArithExpr.sum(five, four)
let diff = ArithExpr.diff(sum, two) func cal(_ expr: ArithExpr) -> Int {
switch expr {
case let .number(value):
return value
case let .sum(left, right):
return cal(left) + cal(right)
case let .diff(left, right):
return cal(left) - cal(right)
}
}
cal(diff) // 输出:7

三、枚举的内存布局

在Swift中查看内存占用大小及对齐方式使用枚举:MemoryLayout

  • size:实际用到的空间大小
  • stride:分配占用的空间大小
  • alignment:内存对齐方式

下面的意思是,Int在内存中占用8个字节,内存对齐数是8:

MemoryLayout<int>.size // 输出:8
MemoryLayout<int>.stride // 输出:8
MemoryLayout<int>.alignment // 输出:8

查看枚举占用内存:

enum Password {
case number(Int, Int, Int, Int)
case other
} var pwd = Password.number(1, 2, 2, 3) MemoryLayout.size(ofValue: pwd) // 输出:33
MemoryLayout.stride(ofValue: pwd) // 输出:40
MemoryLayout.alignment(ofValue: pwd) // 输出:8

为什么是占用内存大小是33,而分配了40?

  • number(Int, Int, Int, Int)占用32个字节,other占用1个字节,所以一共只需要占用33个字节就够用了
  • 因为内存对齐数是8,所以分配内存的时候只能是8的倍数,而33个字节不够8的倍数,所以往高位补齐后就是40了

为什么other占用1个字节呢?

enum Season {
case spring, summer, autumn, winter
}
MemoryLayout<season>.size // 输出:1
MemoryLayout<season>.stride // 输出:1
MemoryLayout<season>.alignment // 输出:1 // 限定类型
enum Season: String {
case spring, summer, autumn, winter
}
MemoryLayout<season>.size // 输出:1
MemoryLayout<season>.stride // 输出:1
MemoryLayout<season>.alignment // 输出:1
  • 上面代码可以看出不管类型是什么占用的内存大小都是1个字节;
  • 其实本质上是关联值和原始值的区别。

结论一: 把传进去的关联值直接存储到枚举变量内存里面的,所以枚举变量是关联值的话,内存是一定和将要存储的关联值大小有关。

为了证实结论一,比较下面的两个不同类型的关联值:

enum Password {
case number(Int, Int, Int, Int)
case other
} MemoryLayout<password>.size // 输出:33
MemoryLayout<password>.stride // 输出:40
MemoryLayout<password>.alignment // 输出:8 enum Password {
case number(String, String, String, String)
case other
} MemoryLayout<password>.size // 输出:65
MemoryLayout<password>.stride // 输出:72
MemoryLayout<password>.alignment // 输出:8

结论二: 原始值固定后是不能修改的,内存中只会把对应的成员值(序号)存下来,这时候1个字节足够用了,和枚举类型无关(不管是Int还是String枚举都是占用一个字节)。

分析下面代码:

enum Season: Int {
// 序号0 序号1 序号2 序号3
case spring = 1, summer = 2, autumn = 3, winter = 4
} var season1 = Season.spring
var season2 = Season.spring
var season3 = Season.spring
MemoryLayout<season>.size // 输出:1
MemoryLayout<season>.stride // 输出:1
MemoryLayout<season>.alignment // 输出:1

疑问:成员值在内存中只占用1个字节,Int或String的原始值是怎么存下的?rawValue其实是另外一块地址。

  • 关联值才会存储到枚举变量中,原始值不会占用枚举变量内存
  • 我们可以通过内存地址看到前面的字节被关联值占用,关联值后面有一个字节是保存成员值
    • 1个字节存储成员值(如果只有一个枚举成员则不占用内存)
    • N个字节存储关联值(N取占用内存最大的关联值),任何一个case的关联值都共用这N个字节(共用体)
    • 剩余字节按照对齐数补齐

Switchcase其实是比较枚举的成员值的

Swift系列四 - 枚举的更多相关文章

  1. 窥探Swift系列博客说明及其Swift版本间更新

    Swift到目前为止仍在更新,每次更新都会推陈出新,一些Swift旧版本中的东西在新Swift中并不适用,而且新版本的Swift会添加新的功能.到目前为止,Swift为2.1版本.去年翻译的Swift ...

  2. Swift 学习- 09 -- 枚举

    // 递归枚举 // 美家居为一组相关的值定义了一个共同的类型, 使你可以在代码中以类型安全的的方式使用这些值. // 如果你熟悉C语言, 你会知道在C语言中, 枚举会为一组整型值分配相关联的名称, ...

  3. Swift 学习笔记 (枚举)

    枚举为一种相关值定义了一个通用类型,从而可以让你在代码中类型安全的操作这些值. Swift中的枚举很灵活,不需要给每一个枚举中的成员都提供值.如果一个值(所谓 原时值) 要被提供给每一个枚举成员,那么 ...

  4. C#中的函数式编程:递归与纯函数(二) 学习ASP.NET Core Razor 编程系列四——Asp.Net Core Razor列表模板页面

    C#中的函数式编程:递归与纯函数(二)   在序言中,我们提到函数式编程的两大特征:无副作用.函数是第一公民.现在,我们先来深入第一个特征:无副作用. 无副作用是通过引用透明(Referential ...

  5. 前端构建大法 Gulp 系列 (四):gulp实战

    前端构建大法 Gulp 系列 (一):为什么需要前端构建 前端构建大法 Gulp 系列 (二):为什么选择gulp 前端构建大法 Gulp 系列 (三):gulp的4个API 让你成为gulp专家 前 ...

  6. Netty4.x中文教程系列(四) 对象传输

    Netty4.x中文教程系列(四)  对象传输 我们在使用netty的过程中肯定会遇到传输对象的情况,Netty4通过ObjectEncoder和ObjectDecoder来支持. 首先我们定义一个U ...

  7. S5PV210开发系列四_uCGUI的移植

    S5PV210开发系列四 uCGUI的移植 象棋小子          1048272975 GUI(图形用户界面)极大地方便了非专业用户的使用,用户无需记忆大量的命令,取而代之的是能够通过窗体.菜单 ...

  8. WCF编程系列(四)配置文件

    WCF编程系列(四)配置文件   .NET应用程序的配置文件 前述示例中Host项目中的App.config以及Client项目中的App.config称为应用程序配置文件,通过该文件配置可控制程序的 ...

  9. SQL Server 2008空间数据应用系列四:基础空间对象与函数应用

    原文:SQL Server 2008空间数据应用系列四:基础空间对象与函数应用 友情提示,您阅读本篇博文的先决条件如下: 1.本文示例基于Microsoft SQL Server 2008 R2调测. ...

随机推荐

  1. CF482E ELCA

    一.题目 点此看题 二.解法 题目的提示已经足够明显了吧,肯定是要写一个 \(\tt link-cut-tree\) .我们只需要求出总和,再除以方案数就是期望.然后可以算每个点为 \(\tt lca ...

  2. 使用 autofac 实现 asp .net core 的属性注入

    使用 autofac 代替 asp .net core 默认的 IOC 容器,可实现属性注入. 之前的使用方式不受影响. 源码已开源: dotnet-campus/Autofac.Annotation ...

  3. 2018ICPC南京 A. Adrien and Austin

    题目: 题意:1-N个石子每次只能取连续的1-K个问输赢.(一开始以为只是个简简单单的巴什游戏,激动的提交了一发wr了,再读了一遍题才发现是只能取连续的) 题解:当n==0或者k==1&&am ...

  4. 【体系结构】Oracle进程架构

    Client Process的介绍 Client and Server Processes Client Process代表着客户端进程,每一个客户端进程关联着一个Server Process(服务器 ...

  5. certutil绕过

    一般进内网过后我都会使用certutil下载文件,但在最近打一台内网机子的时候出现了certutil拒绝访问的情况,在本地搭建了一个环境尝试绕过certutil下载文件. 安装杀软更新到最新版本,开启 ...

  6. SqlServer存储过程的创建与使用

    什么是存储过程? T-SQL中的存储过程,非常类似于net语言中的方法,它可以重复调用.当存储过程执行一次后,可以将语句缓存中,这样下次执行的时候直接使用缓存中的语句. 这样就可以提高存储过程的性能. ...

  7. java例题_47 读取 7 个数(1—50)的整数值,每读取一个值,程序打印出该值个数的*

    1 /*47 [程序 47 打印星号] 2 题目:读取 7 个数(1-50)的整数值,每读取一个值,程序打印出该值个数的*. 3 */ 4 5 /*分析 6 * 1.多次读取---for循环 7 * ...

  8. Oauth2协议那些事

    1. 背景 首先,设想一种情境:你平常会使用一款照片存储App(以下照片服务指代),用来将自己喜欢的照片存放在上面以备随时查看.假如有一天,你想要打印其中的某张照片而且你找到了一款打印照片App(以下 ...

  9. C++中的间接宏函数

    宏函数对于每个C++程序员都决不陌生,就算是初出茅庐的C++程序员也知道如何定义.使用宏函数.   但是当初学者看到类似于以下这种宏函数嵌套的时候,可能还是会比较嘀咕, #define CONVERT ...

  10. SCIP:构造数据抽象--数据结构中队列与树的解释

    现在到了数学抽象中最关键的一步:让我们忘记这些符号所表示的对象.不应该在这里停滞不前,有许多操作可以应用于这些符号,而根本不必考虑它们到底代表着什么东西. --Hermann Weyi <思维的 ...