Swift底层原理探索3----可选项

可选项(Optional)

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

var age: Int?
age = 10
age = nil

C语言中,变量声明之后,会自动被赋予初始值,Swift则不会这样,但是Swift的Optional则会有默认的初始值nil

func test() {
    var age: Int -->没有初始值,必须赋值之后才能使用
    var weight: Double?  --> 默认初始值为nil, 等价于  var weight: Double? = nil
}

使用场景,有可能需要使用nil的地方

var array = [1,13,45,33]
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(5)) // nil


强制解包(Forced Unwrapping)

  • 可选项是对其他类型的一层包装,可以将它理解成一个盒子
    1. 如果为nil,那么它是个空盒子
    2. 如果不为nil,那么盒子里装的是:被包装类型的数据
var age: Int? // 默认就是nil
age = 10
age = nil
image
  • 如果要从可选项中取出被包装的数据(将盒子里装的东西取出来),需要使用感叹号!进行强制解包
var speed: Int? = 10
var speedInt: Int = speed!
speedInt += 10
  • 如果对值为nil的可选项(空盒子)进行强制解包,将会产生运行时错误
var age: Int?
age!

运行报错:Fatal error: Unexpectedly found nil while unwrapping an Optional value: file Optional.xcplaygroundpage


判断可选项是否包含值

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


可选项绑定(Optional Binding)

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

用在枚举值中

enum Season: Int {
    case spring = 1, summer, autumn, winter
}
if let season = Season(rawValue: 6) {  -->有可能得到对应的case, 有可能得不到,那么结果就为空
    switch season {
    case .spring:
        print("the season is spring")
    default:
        print("the season is other")
    }
}  else {
       print("no such season")
}

注意下面的等价用法

if let first = Int("4") {
    if let second = Int("42") {
        if first < second && second < 100 {
            print("\(first) < \(second) < 100")
        }
    }
}

======等价写法如下, 通过逗号来分割 带可选项绑定的 if判断条件

if let first = Int("4"),
    let second = Int("42"),
    first < second && second < 100 {
    print("\(first) < \(second) < 100")
}


while循环中使用可选项绑定

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

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

public func ?? <T>(optional: T?, defaultValue: @autoclosure () throws -> T?) rethrows -> T?
public func ?? <T>(optional: T?, defaultValue: @autoclosure () throws -> T) rethrows -> T

a ?? b

  • a必须是可选项
  • b是可选项 或者 不是可选项
  • ba的存储类型必须相同
  • 如果a不为nil,就返回a
  • 如果anil,就返回b
  • 如果b不是可选项,返回a时回自动解包
let a: Int? = 1
let b: Int? = 2
let c = a ?? b //c是Int?, Optional(1)

let a: Int? = nil
let b: Int? = 2
let c = a ?? b //c是Int?, Optional(2)
let a: Int? = nil
let b: Int? = nil
let c = a ?? b //c是Int?, nil
let a: Int? = 1
let b: Int = 2
let c = a ?? b //c是Int, 1
let a: Int? = nil
let b: Int = 2
let c = a ?? b //c是Int, 2

如果没有??,有些代码写起来就比较麻烦

let a: Int? = nil
let b: Int = 2
let c: Int
if let tmp = a {
    c = tmp
} else {
    c = b
}


多个??一起使用

从左往右算

let a: Int? = 1
let b: Int? = 2
let c = a ?? b ?? 3 //c是Int, 1
let a: Int? = nil
let b: Int? = 2
let c = a ?? b ?? 3 //c是Int, 1
let a: Int? = nil
let b: Int? = nil
let c = a ?? b ?? 3 //c是Int, 1


??跟if let配合使用

let a: Int? = nil
let b: Int? = 2
if let c = a ?? b {
    print(c)
}
-->类似于 if a != nil || b != nil {}
if let c = a, let d = b {
    print(c)
    print(d)
}
-->类似于 if a != nil && b != nil {}


if语句实现登录

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
    }

    print("用户名:\(username)","密码:\(password)", "登录ing")
}

login(["username": "jack", "password": "123456"])
login(["password": "123456"])
login(["username": "jack"])
************运行结果
用户名:jack 密码:123456 登录ing
请输入用户名
请输入密码


guard语句

上面的登录业务通过guard来实现

func login(_ info: [String : String]) {
    guard let username = info["username"] else {
        print("请输入用户名")
        return
    }

    guard let password = info["password"] else {
        print("请输入密码")
        return
    }

    print("用户名:\(username)","密码:\(password)", "登录ing")
}

login(["username": "jack", "password": "123456"])
login(["password": "123456"])
login(["username": "jack"])
***********运行结果
用户名:jack 密码:123456 登录ing
请输入用户名
请输入密码


隐式解包(Implicitly Unwrapped Optional)

  • 在某些情况下,可选项一旦被设定值之后,就会一直拥有值
  • 在这种情况下,可以去掉检查,也不必每次访问的时候都进行解包,因为它能确定每次访问的时候都有值
  • 可以在类型后面加个感叹号!定义一个隐式解包的可选项
let num1: Int! = 10 //如果能确定某个变量会一直不为nil,可以使用这种声明方式,
let num2: Int = num1 // 可以直接使用,系统会自动解包后赋值,当然也可以使用强制解包方式,但是num1的本质还是一个optional
if num1 != nil {
    print(num1 + 6)
}

if let num3 = num1 {
    print(num3)
}


如果给隐式解包类型的optional变量赋值nil,当该变量被赋值给其他变量/常量式,会出现 运行时报错

let num4: Int! = nil
let num5: Int = num4  -->注意,程序执行完这一句才会报错,在执行这一句的时候,会对num4进行隐式解包,结果发现它内部是nil,因此把nil取出来赋值给  num5: Int 的时候就报错了,因为 num5 不能为空
Fatal error: Unexpectedly found nil while implicitly unwrapping an Optional value

上述的案例,让人感觉使用带!的optional存在很多风险,其实用下面的方式莫不是更保险点吗,

var num5: Int = 10
let num5: int = num56

这样至少在程序运行之前,就能看到错误,如果有的话。

实际上,我们大部分场景下,也都是推荐使用带?optional,那用!声明的optional存在的意义是什么呢?

  • 如果你提供一套api给外界使用,并且期望使用者严格遵守你的要求不要传nil过来,并且认为使用者在错误使用的时候而导致程序直接报错崩溃就是你期待的,那么你可以使用这种用法。除此之外,还是不用为妙。


字符串插值

可选项在字符串插值或者直接打印的时候,编译器会发出警告

var age: Int? = 10
print("My age is \(age)")//直接进行字符串插值,会产生编译器警告
//️String interpolation produces a debug description for an optional value; did you mean to make this explicit?️

3种常用的方法可以消除警告

var age: Int? = 10
print("My age is \(age!)")
print("My age is \(String(describing: age))")
print("My age is \(age ?? 0)")


多重可选项

var num1: Int? = 10
var num2: Int?? = num1
var num3: Int?? = 10
print(num2 == num3) // true

num1num2num3的结构分别如下

image

再看如下

var num1: Int? = nil
var num2: Int?? = num1
var num3: Int?? = nil
print(num2 == num3) // false

(num2 ?? 1) ?? 2  //1
(num1 ?? 1) ?? 2  //2
image

可以使用LLDB指令 frame variable -R 或者 fr v -R来查看变量的内部信息。

有关Swift的可选项就整理到这里。

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

推荐阅读更多精彩内容