Swift語法 Swift5 【05 - 可選項】


  • 作者: Liwx
  • 郵箱: 1032282633@qq.com
  • 源碼: 需要源碼的同學, 可以在評論區(qū)留下您的郵箱

iOS Swift 語法 底層原理內存管理分析 專題:【iOS Swift5語法】

00 - 匯編
01 - 基礎語法
02 - 流程控制
03 - 函數(shù)
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,那么盒子里裝的是: 被包裝類型的數(shù)據(jù)

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

  • 如果要從可選項中取出被包裝的數(shù)據(jù)(將盒子里裝的東西取出來), 需要使用感嘆號!進行強制解包

  • 可選項不能直接參與運算

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的可選項(空盒子)進行強制解包,將會產(chǎn)生運行時錯誤
var age: Int?
print(age!)     // 報錯: Fatal error: Unexpectedly found nil while unwrapping an Optional value

03-判斷可選項是否包含值

let number = Int("123")
if number != nil {
    print("字符串轉換整數(shù)成功: \(number!)")  // number!: 強制解包
} else {
    print("字符串轉換整數(shù)失敗")
}
// 字符串轉換整數(shù)成功: 123

04-可選項綁定(Optional Binding)

  • 可以使用可選項綁定來判斷可選項是否包含值
  • 如果包含就自動解包,
    值賦給一個臨時的常量(let)或者變量(var),并返回true,否則返回false
if let number = Int("123") {
     print("字符串轉換整數(shù)成功: \(number)")  // number無需再解包
    // number是強制解包之后的Int值
    // number作用于僅限于這個大括號
} else {
    print("字符串轉換整數(shù)失敗")
}

  • 枚舉可選項綁定
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循環(huán)中使用可選項綁定

  • 遍歷數(shù)組,將遇到的正數(shù)都加起來,如果遇到負數(shù)或者非數(shù)字,停止遍歷
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)

  • 空合并運算符??在標準庫中的定義
    • ?? 第一個(前面)參數(shù)必須是可選項
    • ?? 返回值類型取決于 ?? 后面參數(shù)的類型
// ??后面參數(shù)類型為T?, 則返回值類型為T?
public func ?? <T>(optional: T?, defaultValue: @autoclosure
() throws -> T?) rethrows -> T?

// ??后面參數(shù)類型為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-多個?? 一起使用

  • 多個??一起使用時, 返回值類型取決于最后一個參數(shù)的類型

  • 示例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語句實現(xiàn)登陸

  • 使用if-let實現(xiàn)登陸
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時,就會執(zhí)行大括號里面的代碼
  • 當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-字符串插值

  • 可選項在字符串插值或者直接打印時,編譯器會發(fā)出警告(在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 查看區(qū)別

  • 示例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 - 枚舉


最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
禁止轉載尸饺,如需轉載請通過簡信或評論聯(lián)系作者条摸。
  • 序言:七十年代末剿骨,一起剝皮案震驚了整個濱河市辰斋,隨后出現(xiàn)的幾起案子策州,更是在濱河造成了極大的恐慌,老刑警劉巖宫仗,帶你破解...
    沈念sama閱讀 206,839評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件够挂,死亡現(xiàn)場離奇詭異,居然都是意外死亡藕夫,警方通過查閱死者的電腦和手機下硕,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來汁胆,“玉大人梭姓,你說我怎么就攤上這事∧勐耄” “怎么了誉尖?”我有些...
    開封第一講書人閱讀 153,116評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長铸题。 經(jīng)常有香客問我铡恕,道長,這世上最難降的妖魔是什么丢间? 我笑而不...
    開封第一講書人閱讀 55,371評論 1 279
  • 正文 為了忘掉前任探熔,我火速辦了婚禮,結果婚禮上烘挫,老公的妹妹穿的比我還像新娘诀艰。我一直安慰自己,他們只是感情好饮六,可當我...
    茶點故事閱讀 64,384評論 5 374
  • 文/花漫 我一把揭開白布其垄。 她就那樣靜靜地躺著,像睡著了一般卤橄。 火紅的嫁衣襯著肌膚如雪绿满。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,111評論 1 285
  • 那天窟扑,我揣著相機與錄音喇颁,去河邊找鬼漏健。 笑死,一個胖子當著我的面吹牛橘霎,可吹牛的內容都是我干的蔫浆。 我是一名探鬼主播,決...
    沈念sama閱讀 38,416評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼茎毁,長吁一口氣:“原來是場噩夢啊……” “哼克懊!你這毒婦竟也來了忱辅?” 一聲冷哼從身側響起七蜘,我...
    開封第一講書人閱讀 37,053評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎墙懂,沒想到半個月后橡卤,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,558評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡损搬,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,007評論 2 325
  • 正文 我和宋清朗相戀三年碧库,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片巧勤。...
    茶點故事閱讀 38,117評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡嵌灰,死狀恐怖,靈堂內的尸體忽然破棺而出颅悉,到底是詐尸還是另有隱情沽瞭,我是刑警寧澤,帶...
    沈念sama閱讀 33,756評論 4 324
  • 正文 年R本政府宣布剩瓶,位于F島的核電站驹溃,受9級特大地震影響,放射性物質發(fā)生泄漏延曙。R本人自食惡果不足惜豌鹤,卻給世界環(huán)境...
    茶點故事閱讀 39,324評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望枝缔。 院中可真熱鬧布疙,春花似錦、人聲如沸愿卸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,315評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽擦酌。三九已至俱诸,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間赊舶,已是汗流浹背睁搭。 一陣腳步聲響...
    開封第一講書人閱讀 31,539評論 1 262
  • 我被黑心中介騙來泰國打工赶诊, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人园骆。 一個月前我還...
    沈念sama閱讀 45,578評論 2 355
  • 正文 我出身青樓舔痪,卻偏偏與公主長得像,于是被迫代替她去往敵國和親锌唾。 傳聞我的和親對象是個殘疾皇子锄码,可洞房花燭夜當晚...
    茶點故事閱讀 42,877評論 2 345