前言:
本篇仅为视频学习笔记
空合并运算符 ?? (Nil-Coalescing Operator)
上图是swift对空合并运算符的定义。
★ a ?? b
**条件:** - a 是可选项,不能 10 ?? 20这样写 - 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)
如上代码,a和b都是可选项,所以可以 a??b 这样写,那刚才刚说过a??b这个家伙会返回什么,所以先看a是不是nil,很明显a不是nil,所以它返回a,所以它直接将a这个可选项赋值给c。所以c最终就是a的值。那么c是什么类型呢?是刚刚返回的a,a是Int?类型,所以c也是Int?类型。
所以最终结果是,c是Int?类型,并且里面包装的值是1.如// c是Int?,Optional(1)。
例-2
那么,再看另一个问题,如果向下面代码那样写呢?
let a: Int? = nil let b: Int? = 2 let c = a ?? b // c是Int?,Optional(2)
首先,这两个家伙都是可选项,然后呢,a的值是nil。那么,a的值是nil,那么就返回b,那么b是Int?类型,那么你直接赋值给c。那么c也是Int?类型,并且它里面包装的是2。如: // c是Int?,Optional(2)。这个没有什么疑问。
例-3
let a: Int? = nil let b: Int? = nil let c = a ?? b // c是Int?,nil
再看另一个问题,如上代码,如果两个常量a,b,都为nil。首先,如果a为nil,那就返回b,那就将b赋值给c。b是nil,实际上就是把nil赋值给了c。既然c为nil,那必然也是Int?类型的
例-4
let a: Int? = 1 let b: Int = 2 let c = a ?? b // c是Int,1
上述代码,a依然是一个可选项,b呢,它不是可选项。那么,这里面会怎么做呢?如果a不为nil,就返回a。是这样,但是之前说过这样一个问题,如果右边这个b它是非可选项,那么你再返回a的时候,要对a进行一个解包。所以,a它会自动解包,相当于a不为nil,返回a的时候,它会将a解包,相当于a后边加一个感叹号一样,就相当于将1这个Int类型的值赋值给了c。所以到时候c存储的是Int类型的,也就是1,而不是Int?,注意这个问题,所以 // c是Int,1。
综上所述,a返回的类型取决于b,如果b是可选类型,那么这个运算符返回的也是可选类型,如果这个b是非可选类型,那么这个运算符??返回的也是非可选类型。所以返回的类型取决于b。
小结
怎么证明a返回的类型取决于b?
a返回的类型取决于b,其实是可以证明这个问题的。我们观察一下swift源码里面对??运算符的定义,如上图,这个运算符的话,他是接受两个参数,第一个参数要求是什么?T?问号类型,按照刚才前面说的??问号前面的这个值必须是可选项,第二个参数,defaultValue,可能大家还看不懂先不用管。但是如下图:
我们看对第二个参数的要求,可以是可选类型,可以是非可选类型。我们在观察它的返回值。
返回值的类型,和最后一个参数的类型保持一致。你最后一个参数带问号,返回值也是带问号。你最后一个参数不带问号,那么你的返回值也是不带问号的。
多个??一起使用
那么,我们再来看,多个空合并运算符一起使用,举个例子如下:
let a: Int = 1 let b: Int? = 2 let c = a ?? b ?? 3 // c是Int, 1
如上面 a ?? b ?? 3 ,这个。如果这么做的话那么它返回的到底是什么类型呢?聪明的人,可能一眼就能看出来,最右边。因为最右边是3那也就意味着c肯定是一个Int类型,而不是int?类型。这就说明了如果很多空合并运算符在一起的话,要知道最终返回的类型是什么,我们就看最后一个运算符的值是什么类型的,就知道了。
我们一点点算的话,也是可以的,从左到右边,我们看这个a??b,返回a啊。这个是怎么做的,首先看a为不为nil,很明显a是不为nil的,所以返回a。而且最右边这个b,它是可选类型,所以返回的a是可选类型,那么就变成了a??3.
那么,a??3,a是不为nil的,既然它不为nil,所以返回a,但是由于右边3是Int类型,所以这个a会解包,将1解包出来,最终将1赋值给c,所以到时候,c就是什么?Int类型。下图如上:
那么,会有一个问题。如下:可不可以这么做呢?
var a: Int? = 1 var b: Int? = 2 var c = a ?? 3 ?? b print(c)
如上图,我们先看一下c是什么类型,是Int?类型。
因为我们按照这个运算符定义的话,看它做右边,那么最右边它是Int?所以c它也是Int?但是你明显发现会有警告建议你不要这么做。
为什么呢?因为我们两个??的这个运算符,b前边的这个家伙必须是可选类型才行,如定义介绍。但是,你思考一下,由于你这个是a??3,所以到时候,肯定是看3这个类型,所以a??3,返回的肯定是Int类型,就相当于你把Int类型放在了左边。如下图:
它肯定会给你警告啊,所以建议不要这么做。
??跟 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 ?? b,由于a和b是可选类型,如果我想取它们其中一个有值的家伙并且解包出来该怎么做呢?就可以if let c = a ?? b这么做。
那么,a和b是可以选类型,那么可选类型,我们要干嘛,肯定要进行一个可选型类型绑定,如果绑定成功,你就可以自动解包print(c)。就把你的值取出来。所以思考一下,我们这种写法你不觉得有点类似于if a != nil || b != nil 吗?