Swift语法 Swift5 【05 - 可选项】


  • 作者: Liwx
  • 邮箱: 1032282633@qq.com
  • 源码: 需要源码的同学, 可以在评论区留下您的邮箱

iOS Swift 语法 底层原理内存管理分析 专题:【iOS Swift5语法】

00 - 汇编
01 - 基础语法
02 - 流程控制
03 - 函数
04 - 枚举
05 - 可选项
06 - 结构体和类
07 - 闭包
08 - 属性
09 - 方法
10 - 下标
11 - 继承
12 - 初始化器init
13 - 可选项


01-可选项(Optional)

  • 可选项,一般也叫可选类型,它允许将值设置为nil
  • 类型名称后面加个问号?来定义一个可选项
var name: String? = "Jack"
name = nil

var age: Int?   // 默认就是nil
age = 10  // Optional(10)
age = nil
  • 可选项的简单使用
var array = [1, 15, 40, 29]
func get(_ index: Int) -> Int? {
    if index < 0 || index >= array.count {
        return nil
    }
    return array[index]
}
print(get(1))   // Optional(15)
print(get(-1))  // nil
print(get(4))   // nil

02-强制解包

  • 可选项是对其他类型的一层包装,可以将它理解为一个盒子
  • 如果为nil,那么他是一个空盒子
  • 如果不为nil,那么盒子里装的是: 被包装类型的数据

var age: Int?   // 默认就是nil
age = 10
age = nil
QQ20200420-154654.png

  • 如果要从可选项中取出被包装的数据(将盒子里装的东西取出来), 需要使用感叹号!进行强制解包

  • 可选项不能直接参与运算

var age: Int? = 10

// 可选项不能直接参与运算
//var num = age + 10  // error: value of optional type 'Int?' must be unwrapped to a value of type 'Int'
var num = age! + 20

print(age)      // Optional(10)
print(num)      // 30

  • 如果对值为nil的可选项(空盒子)进行强制解包,将会产生运行时错误
var age: Int?
print(age!)     // 报错: Fatal error: Unexpectedly found nil while unwrapping an Optional value

03-判断可选项是否包含值

let number = Int("123")
if number != nil {
    print("字符串转换整数成功: \(number!)")  // number!: 强制解包
} else {
    print("字符串转换整数失败")
}
// 字符串转换整数成功: 123

04-可选项绑定(Optional Binding)

  • 可以使用可选项绑定来判断可选项是否包含值
  • 如果包含就自动解包,
    值赋给一个临时的常量(let)或者变量(var),并返回true,否则返回false
if let number = Int("123") {
     print("字符串转换整数成功: \(number)")  // number无需再解包
    // number是强制解包之后的Int值
    // number作用于仅限于这个大括号
} else {
    print("字符串转换整数失败")
}

  • 枚举可选项绑定
enum Season : Int {
    case spring = 1, summer, autumn, winter
}

if let season = Season(rawValue: 6) {
    switch season {
    case .spring:
        print("the season is spring")
    default:
        print("the season is other")
    }
} else {
    print("no such season")
}
// no such season

05-等价写法

  • 可选项绑定,多个条件要同时成立,用逗号隔开
// 写法1
if let first = Int("4") {
    if let second = Int("42") {
        if first < second && second < 100 {
            print("\(first) < \(second) < 100")
        }
    }
}
// 4 < 42 < 100

// - 可选项绑定,多个条件要同时成立,用`逗号`隔开
// 写法2 推荐
if let first = Int("4"),
    let second = Int("42"),
    first < second && second < 100 {
    print("\(first) < \(second) < 100")
}
// 4 < 42 < 100

注意: 不允许 可选项绑定&&一起使用

if let s = Season(rawValue: 2) && age > 10 {

}

06-while循环中使用可选项绑定

  • 遍历数组,将遇到的正数都加起来,如果遇到负数或者非数字,停止遍历
var strs = ["10", "20", "abc", "-20", "30"]
var index = 0
var sum = 0
while let num = Int(strs[index]), num > 0 {
    sum += num
    index += 1
}
print(sum)  // 30

07-空合并运算符??(Nil-Coalescing Operator)

  • 空合并运算符??在标准库中的定义
    • ?? 第一个(前面)参数必须是可选项
    • ?? 返回值类型取决于 ?? 后面参数的类型
// ??后面参数类型为T?, 则返回值类型为T?
public func ?? <T>(optional: T?, defaultValue: @autoclosure
() throws -> T?) rethrows -> T?

// ??后面参数类型为T, 则返回值类型为T
public func ?? <T>(optional: T?, defaultValue: @autoclosure
() throws -> T) rethrows -> T

  • 空合并运算 a ?? b
    • a 必须是可选项
    • b 可选项 或者 不是可选项
    • b 跟a的存储类型必须相同
    • 如果a不为nil,就返回a
    • 如果a为nil,就返回b
    • 如果b不是可选项, 返回a是会自动解包

  • 示例1
let a: Int? = 1
let b: Int? = 2
let c = a ?? b  // c是Int?, Optional(1)
print(c)  // Optional(1)
  • 示例2
let a: Int? = nil
let b: Int? = 2
let c = a ?? b  // c是Int?, Optional(2)
print(c)  // Optional(2)
  • 示例3
let a: Int? = nil
let b: Int? = nil
let c = a ?? b  // c是Int?, nil
print(c)  // nil
  • 示例4
let a: Int? = 1
let b: Int = 2
let c = a ?? b  // c是Int, 1
print(c)  // 1
  • 示例5
let a: Int? = nil
let b: Int = 2
let c = a ?? b  // c是Int, 2
print(c)

  • 对比不使用??运算符
let a: Int? = nil
let b: Int = 2
// 如果不使用??运算符
let c: Int
if let tmp = a {
    c = tmp
} else {
    c = b
}

08-多个?? 一起使用

  • 多个??一起使用时, 返回值类型取决于最后一个参数的类型

  • 示例1

let a: Int? = 1
let b: Int? = 2
let c = a ?? b ?? 3 // c是Int, 1  
print(c)
  • 示例2
let a: Int? = nil
let b: Int? = 2
let c = a ?? b ?? 3
print(c)  // c是Int, 2
  • 示例3
let a: Int? = nil
let b: Int? = nil
let c = a ?? b ?? 3 // c是Int, 3
print(c)

09-??跟if let配合使用

  • 示例1
let a: Int? = nil
let b: Int? = 2
// 类似于if a != nil || b != nil
if let c = a ?? b {
    print(c)    // 2
}
  • 示例2
let a: Int? = 1
let b: Int? = 2
// 类似于if a != nil && b != nil
if let c = a, let d = b {
    print("c:", c, "d:", d)  // c: 1 d: 2
}

10-if-let语句实现登陆

  • 使用if-let实现登陆
func login(_ info: [String : String]) {
    
    let userName: String
    if let tmp = info["userName"] {
        userName = tmp
    } else {
        print("请输入用户名")
        return
    }
    
    let password: String
    if let tmp = info["password"] {
        password = tmp
    } else {
        print("请输入密码")
        return
    }
    
    // if userName ....
    // if password ....
    print("用户名: \(userName)", "密码: \(password)")
}
login(["userName": "liwx", "password": "123456"])   // 用户名: liwx 密码: 123456
login(["password": "123456"])   // 请输入用户名
login(["userName": "liwx"])     // 请输入密码

11-guard语句

  • guard语句的条件为false时,就会执行大括号里面的代码
  • 当guard语句的条件为true时,就会跳过guard语句
  • guard语句特别适合用来"提前退出"

  • guard 格式
guard 条件 else {
    // do something ...
    退出当前作用域
    // return、break、continue、throw error
}

  • 当使用guard语句进行可选项绑定时,绑定的常量(let)变量(var)也能在外层作用域中使用
func login(_ info: [String : String]) {
    
    // 使用guard语句进行可选项绑定
    // userName和password 可以在外层作用域中使用
    guard let userName = info["userName"] else {
        print("请输入用户名")
        return
    }
    guard let password = info["password"] else {
        print("请输入密码")
        return
    }
    
    // if userName ....
    // if password ....
    print("用户名: \(userName)", "密码: \(password)")
}
login(["userName": "liwx", "password": "123456"])

12-隐式解包(Implicitly Unwrapped Optional)

  • 在某些情况下,可选项一旦被设定值之后,就会一直拥有值
  • 在这种情况下,可以去掉检查,也不必每次访问的时候都进行解包,因为它能确定每次访问的时候都有值
  • 可以再类型后面加个感叹号!定义一个隐式解包的可选项

  • 隐式解包可选项的值可以直接运算
let num1: Int! = 10
let num2: Int = num1
if num1 != nil {
    // 隐式解包可选项的值可以直接运算
    print(num1 + 6) // 16
}

  • 隐式解包可选项也可以进行可选项绑定
if let num3 = num1 {
    print(num3)     // 10
}
  • 隐式解包可选项如果没有值(值为nil), 运行时会报错
// Fatal error: Unexpectedly found nil while implicitly unwrapping an Optional value
//let num4: Int! = nil
//let num5: Int = num4

13-字符串插值

  • 可选项在字符串插值或者直接打印时,编译器会发出警告(在Playground没有警告)
var age: Int? = 10
print("My age is \(age)")

// 至少有3种方法消除警告
print("My age is \(age!)")
print("My age is \(String(describing: age))")
print("My age is \(age ?? 0)")

14-多重可选项

  • 使用lldb指令 frame variable -R 或者 fr v -R 查看区别

  • 示例1
var num1: Int? = 10
var num2: Int?? = num1
var num3: Int?? = 10
print(num2 == num3) // true 断点观察
(lldb) fr v -R num1
(Swift.Optional<Swift.Int>) num1 = some {
  some = {
    _value = 10
  }
}
(lldb) fr v -R num2
(Swift.Optional<Swift.Optional<Swift.Int>>) num2 = some {
  some = some {
    some = {
      _value = 10
    }
  }
}
(lldb) fr v -R num3
(Swift.Optional<Swift.Optional<Swift.Int>>) num3 = some {
  some = some {
    some = {
      _value = 10
    }
  }
}
image.png

  • 示例2
var num1: Int? = nil
var num2: Int?? = num1
var num3: Int?? = nil
print(num2 == num3) // false
print(num1 == num3) // false
(lldb) fr v -R num1
(Swift.Optional<Swift.Int>) num1 = none {
  some = {
    _value = 0
  }
}
(lldb) fr v -R num2
(Swift.Optional<Swift.Optional<Swift.Int>>) num2 = some {
  some = none {
    some = {
      _value = 0
    }
  }
}
(lldb) fr v -R num3
(Swift.Optional<Swift.Optional<Swift.Int>>) num3 = none {
  some = some {
    some = {
      _value = 0
    }
  }
}
image.png

iOS Swift 语法 底层原理内存管理分析 专题:【iOS Swift5语法】

下一篇: 06 - 结构体和类
上一篇: 04 - 枚举


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,372评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,368评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,415评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,157评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,171评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,125评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,028评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,887评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,310评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,533评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,690评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,411评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,004评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,812评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,693评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,577评论 2 353