聊聊swift语言中的“??”

最近很多同学问我,swift语言中,??是怎么回事。因为在微信交流中,问题不能被清晰表述,所以我很奇怪对于这么简单的一个运算符,会有这么多同学有疑问。后来随着对问题理解的深入,我渐渐意识到了大家遇到的??的问题和我想象的是不一样的。正巧,对于??,就算只是简单地把它当做运算符看待,也有很多值得挖掘的地方,为此,便有了这篇文章,绝对深入地来探讨一下:

  1. 为什么要有?? -- 空合运算符
  2. swift实现空合运算符的背后
  3. ??仅仅是一个运算符吗?

众所周知,可选型(Optional)是swift语言的一个很重要的语言特性。窃以为,从swift1到现在的swift2.2,很多改变都直接或者间接地和可选型相关。其中,在swift2中“偷偷”引入了一个看起来很稀疏平常的新运算符—— ??,英文名称为Nil Coalescing Operator

最新的swift文档的官方中文翻译中,??被翻译成了空合运算符。如果我没有记错,在第一版的官方翻译中,将其翻译成nil聚合运算符。其实,“聚合”二字属于直译,正是coalesce(读作/kəʊəˈlɛs/)一词的意义,现在的翻译“空合”,将nil的含义——“空”,也放在了其中。值得一提的是,空合运算并非swift语言的首创,C#PerlPHP7.0.0等等均有这个运算符(或者叫Null Coalescing Operator

空合运算符做的事情很简单。x ?? y表示判断x是否为nil,若不为nil,则将x解包后返回,否则,取y的值。比如下面的代码:

let username = loginName ?? "Guest"

假设有一个“系统”,用户可以实名登录,也可作为游客匿名登录,此时,用户的登录名loginName则应该表示为可选型。而系统最终显示的用户名username则需要根据loginName的内容做一个判断,如果loginName为空,则使用默认名称"Guest"

好了,大多数教程讲到这里就已经讲完了空合运算符。但是对于我们这群超酷的swifter而言,这一切仅仅是开始:)

1. 为什么要有空合运算符?

首先,一个显而易见的问题是,这样写和下面两种写法,除了代码更简洁以外,还有其他区别吗?

// 写法1,使用if-else
let username: String
if let loginName = loginName{
    username = loginName
}
else{
    username = "Guest"
}

// 写法2,使用三目运算符(ternary operator)
let username = loginName != nil ? loginName! : "Guest" 

恩,在我们的这个例子中,答案是没有区别??在这里的使用就只是让代码更加简洁了。但要知道,如果我们只是因为代码的简洁就给语言添加各种奇怪的符号,这种语言势必可读性很差。那么多语言都引入空合运算符,绝不仅仅只是因为代码简化,难道空合运算还有功能作用?

相信很多朋友早就感觉到了,空合运算x ?? y像极了一种三目运算符使用的简化形式。三目运算符的使用是这样的:condition ? x : y。我们需要判断表达式condition的真假,若为真,则取x,否则取y。而空合运算就是在判断可选型是否为nil的情况下,一种特殊的使用三目运算的形式:x != nil ? x : y。因为在很多语言中,空可以和布尔值False直接对应,所以我们抽象地将其称为x ? x : y

为什么这种特殊的使用三目运算符的方式要被一个新的符号??所取代?答案就蕴藏在x ? x : y这个式子中。这里注意,x被使用了两次!当我们的x仅仅是一个变量的时候,这无关痛痒。但如果x是一套复杂的逻辑呢?

依然以我们引以为傲,宣称自由民主的可以匿名登录的系统为例。现在登录过程有一个函数login(),返回String?,登陆成功则返回登录用户名的String,失败则返回nil。那么此时,使用三目运算符获取系统显示的用户名就变成了这个样子:

let username = login() != nil ? login()! : "Guest" 

有没有觉得整个人都不好了?为什么要登录(调用login())两次?在这里如果只是对效率有影响也就罢了(也!!就!!罢!!了!!你的年终奖应该泡汤了!)更关键的是,如果login()函数带有副效果(side effect)怎么办?比如,用户登录成功的话,顺便给一个记录用户登陆次数的指示器+1,此时,逻辑都出问题了,用户的每次登录被记录了两次!

空合运算符因此应运而生。相较于x ? x : y,在x ?? y中,保证了x只被运算了一次。这便是空合运算符的功能作用。

let username = login() ?? "Guest" 

当然了,很多同学都表示不会写出login() != nil ? login()! : "Guest"这么傻的代码。但是,要知道,在这里我只是举一个例子。对于login这么重要的模块儿,我们当然不会这么做。但是,你真的敢保证,自己在庞大的工程深处,没有写出过类似性质的代码吗?哦,你敢保证?我的意思是,你敢保证你的队友,不是,对手公司派来的内奸,也不会写出这样的代码吗?:)

2. 空合运算符的背后

你以为空合运算背后真的这么简单吗?现在让我们在XCode中,按住Command点击这对儿可爱的??,看看这个运算符在swift标准库中是怎么声明的?答案是这样的:

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

??左边的那个变量叫optional,是一个T?类型,右边那个变量叫defaultValue,意思是默认值。恩,很好,非常可以理解,optional是可选型,万一是nil我们就用默认值(defaultValue)。这和我们的理解完全一样,so far so good。但是,等等,defaultValue的类型是什么鬼?() throws -> T,这俨然是个函数啊?可我们在使用的时候一直在传入一个变量值甚至是硬编码的常量值啊?在defalutValue 前的那个@autoclosure又是什么鬼?

让我静静

事实上,之前,我们一直在探讨x ?? y这个模型里的x,但是y呢?

回到我们自由民主的伟大系统中,为了昭显这个系统的伟大,对于游客,我们的系统才不会使用千篇一律的"Guest"这个名字呢,而是使用世界一流的智能算法,为每一名游客起一个好听到没朋友的名字。恩恩,这个功能被封装在了一个看起来很平庸的函数中——generateGuestName()。我们就是要这么低调。虽然这个函数里面并不低调,它使用了最先进的智能算法,实时采集了全世界最酷的名字(包括狗的名字,当然,这属于机密,不能外泄),然后随机选择了一个...恩,总之很复杂,消耗计算资源很多就对了。

现在,我们的系统为了显示用户名,就要这样:

let username = login() ?? generateGuestName() 

这么写有什么问题?如果不经过特殊处理,编译器会先执行login(),之后不管login()返回值是不是nil,都再调用一下generateGuestName()!不信,我们自己试验一下。照着??的声明,实现一个我们自己的空合运算符,管它叫???吧:

infix operator ???{}
func ???<T>(optional: T? , defaultValue: T) -> T{   
    if let value = optional{ return value }
    return defaultValue
}

之后,我们这样尝试使用一下我们自己的 ???:

func A() -> String{
    print("A is called!!!")
    return "A"
}

func B() -> String{
    print("B is called!!!")
    return "B"
}

let AorB = A() ??? B()

// 控制台打印出
// A is called!!!
// B is called!!!

看看控制台!Oh My Gosh,B is called被打印了出来!尽管A()函数没有任何质疑的为我们返回了一个非可选型的字符串,B()函数依然被执行了!

回到我们的登录系统,如果我们不做些什么,login() ?? generateGuestName() 这个式子将无视你是否有真实的名字,都要强行调用一下generateGuestName()。我们的系统看起来不是那么自由民主了。(当然,其实我只是从效率角度思考问题。毕竟这关系到我的年终奖😅)

怎么解决这个问题?答案是让defaultValue不是一个值,而是一个函数,这样一来,我们可以在需要的时候再执行这个函数,而不需要的话,就不执行。我们把自己的???这样改写:

// 注意defaultValue的定义,传入的是一个函数参数!
func ???<T>(optional: T? , defaultValue: ()->T) -> T{
    
    if let value = optional{ return value }
    return defaultValue() 
    //注意defaultValue后面的小括号,这是一次函数执行。
    //在此之前,defaultValue不会被计算!
}

// A(), B()两个函数定义和前面一样,不再重复

let AorB = A() ??? B  
// 注意,此时,???的第二个参数是传入一个函数

// 控制台打印出
// A is called!!!
// 并且只有A is called!!! 欧耶!

非常棒!这样我们解决了一个很重要的世界难题。我没有夸张,听说过最短路原则吗?我们为自己的???符号添加上了执行最短路原则的能力!像伟大的&&运算符和||运算符一样,只有在需要的时候,我们才会计算第二个操作符的值!

不过,这样一来,缺点也很明显。一方面,A() ??? B这个调用形式很奇怪(竟然不对称?这是要逼死处女座啊!?),另一方面,我们的???的第二个参数只能传入函数型参数,为了传入一个值,我们必须手动将其封装成一个函数,当然,最简单的封装方式是做成一个闭包,像这样:

let AorX = A() ??? {return "X"}

这未免太不方便了:(

为此,苹果向我们提供了一个新的关键字,@autoclosure,看名字就知道了,这个关键字将一个类型自动转换成了一个闭包。如果你本身就是闭包(或者函数),非常好,你将享受这种延迟调用带来的优势;但如果你不是闭包,你将被自动封装成闭包,同时也享受了这种延迟调用带来的性能提升!比如"X",就会被封装成{return "X"}

试试看:

infix operator ???{}
func ???<T>(optional: T? , @autoclosure defaultValue: ()->T) -> T{
    if let value = optional{ return value}
    return defaultValue()
}

let AorB = A() ??? B()
let AorX = A() ??? "X"

以上两个调用都是合法的,并且在第一个调用中,只有A()被执行了,B()不会被执行。使用@autoclosure,你将之前的两个问题都解决了。又可以愉快地和处女座的同学做朋友了:)

值得一提的是,此时,你已经可以扔掉自己定义的???而更加自信地去使用系统提供给你的??了。我们的???和系统的??逻辑上是一模一样的,只不过系统的??还提供了错误处理的功能(throws and rethrows)。以后有机会,我再向大家介绍throwsrethrows里好玩儿的东东,包括@autoclosure相关,还有很多值得我们探讨的内容。恩,最后的这句话是广告,不过你已经看完了:)

3. ?? 仅仅是一个运算符吗?

到目前为止,我们一直在分析空合运算符。但要注意,我的文章标题不是“聊聊空合运算符”,而是“聊聊??”。在swift语言中,??还能另作他用?

随便创建一个ios的项目,然后随便找一个可以执行的地方写这么一行代码:UIApplication.sharedApplication().delegate?.window,看看这个window的类型,你发现了什么?

我就是专门迷惑swift程序员的"??"类型

怎么类型是UIWindow??,两个问号是怎么个意思?空合?空在哪里?怎么合啊!?还能不能愉快地玩耍了?老子不干了!

别急,我们冷静下来,分析分析。在这里,??显然是修饰UIWindow的(咦?怎么像英语语法课?恩,语法课都差不多啦,这里是swift语法课...)就像?用于修饰UIWindow,就构成了UIWindow的可选型UIWindow?一样。为了方便,我们下面举例用大家都喜闻乐见的Int,我们可以这么声明一个类型:

var x: Int??

编译器静悄悄地让这句声明通过了编译,连一丁点儿抵抗的意思都没有。这个神秘的x是什么类型?

不要想得太复杂,这里没有任何神秘的元素,既然?表示可选型,这个??表示的就是可选型的可选型。换句话说,我们的变量x中,可以存放nil,也可以存放一个整型的可选型Int?

这么理解是不是清晰了很多?也就是说,如果x里要真的存放了一个整数的话,我们需要解包两次!比如这样:

var x: Int?? = 42
if let anotherX = x{
    // 此时,anotherX的类型是整型的可选型Int?
    // 要想使用其中的值,还是需要解包!
    if let value = anotherX{
        print(value)
    }
}
// 控制台打印出42

当然了,我们也可以这么解包。这里,希望你没有被太多的x搞晕:

if let x = x, value = x{
    print(value)
}
// 控制台打印出42

不过看起来这么欠揍的x,很多同学可能更愿意硬来...

print(x!!)  // 注意,是两个叹号!!
// 控制台打印出42

恩,理解了这个可选型的可选型??以后,相信脑洞大开的各位swifter一定会满心欢喜地写出这样的声明:

let realX: Int???????????????? = 42

编译器是不会报错的,一切都合法。下次迷惑对手的时候就这么干。

let realX: Int???????????????? = 42

不过言归正传,??除了迷惑对手以外,有什么实际意义?毕竟我们作为闷声发大财...哦,不对,闷声研究计算机科学的swifter来说,没有那么多对手啊...

实际上,对于我们之前的例子而言,UIApplication.sharedApplication().delegate下的window类型是UIWindow??,这背后是两个机制共同作用的结果。我们首先需要看一下这个UIApplication.sharedApplication().delegate是什么东东?按住Command点击delegate,我们可以看到,这个delegate是定义在UIApplication下的一个UIApplicationDelegate?类型的成员变量:

public class UIApplication : UIResponder {
    public class func sharedApplication() -> UIApplication
    unowned(unsafe) public var delegate: UIApplicationDelegate?
    // ...
    // 以下省略很多代码
}

按住Command看一下UIApplicationDelegate的定义,就会发现,UIApplicationDelegate是一个协议,里面的window这个变量是这么定义的:

public protocol UIApplicationDelegate : NSObjectProtocol {
    // ... 省略了很多代码 
   
    @available(iOS 5.0, *)
    optional public var window: UIWindow? { get set }
    // XCode7.3中,在UIApplication.h的395行

    // ... 省略了很多代码
}

UIApplicationDelegate是一个协议。在swift中,协议中定义的属性或者方法如果想被表示为可选择被实现的,需要加上optional的关键字(还要声明成@objc,有时间再深入介绍啦)。相应的,在协议的实现中,这个“可选择被实现的属性或者方法”,就会被解析成可选型。这是第一个?

另一个层面,在这个协议的定义里,window变量本身就被定义成一个可选型UIWindow?,于是有了第二个?

也就是说,第一个?表示这个变量在协议中是可选择实现的;第二个?,表示这个变量可以为nil。这便是我们的例子里,最终实现出来供我们使用的UIApplication.sharedApplication().delegate?.window,其类型是UIWindow??的原因。

但在我们实际编程中,会经常使用诸如Int??这种类型吗?反正我还没有遇到过。如果大家发现了需要Int??才能合理表意的场景,请一定和我分享:)


版权声明:liuyubobobo原创作品。仅允许对原始文章链接进行转发传播。如需转载,请联系原文作者:liuyubobobo@gmail.com。未经允许转载,必追究法律责任。

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

推荐阅读更多精彩内容