当前位置: 首页 > 编程笔记 >

Swift中使用可选类型完美解决占位问题

常自怡
2023-03-14
本文向大家介绍Swift中使用可选类型完美解决占位问题,包括了Swift中使用可选类型完美解决占位问题的使用技巧和注意事项,需要的朋友参考一下

可选类型是Swift中新引入的,功能很强大。在这篇博文里讨论的,是在Swift里,如何通过可选类型来保证强类型的安全性。作为例子,我们来创建一个Objective-C API的Swift版本,但实际上Swift本身并不需要这样的API。

为Dictionary增加objectsForKeys函数

在Objective-C中,NSDictionary有一个方法-objectsForKeys:NoFoundMarker:, 这个方法需要一个NSArray数组作为键值参数,然后返回一个包含相关值的数组。文档里写到:"返回数组中的第N个值,和输入数组中的第N个值相对应",那如果有某个键值在字典里不存在呢?于是就有了notFoundMarker作为返回提示。比如第三个键值没有找到,那么在返回数组中第三个值就是这个notFoundMarker,而不是字典中的第三个值,但是这个值只是用来提醒原字典中没有找到对应值,但在返回数组中该元素存在,且用notFoundMarker作为占位符,因为这个对象不能直接使用,所以在Foundation框架中有个专门的类处理这个情况:NSNull。

在Swift中,Dictionary类没有类似objectsForKeys的函数,为了说明问题,我们动手加一个,并且使其成为操作字典值的通用方法。我们可以用extension来实现:


extension Dictionary{

    func valuesForKeys(keys:[K], notFoundMarker: V )->[V]{

        //具体实现代码后面会写到

    }

}


以上就是我们实现的Swift版本,这个和Objective-C版本有很大区别。在Swift中,因为其强类型的原因限制了返回的结果数组只能包含单一类型的元素,所以我们不能放NSNull在字符串数组中,但是,Swift有更好的选择,我们可以返回一个可选类型数据。我们所有的值都封包在可选类型中,而不是NSNull, 我们只用nil就可以了。

extension Dictionary{

    func valuesForKeys(keys: [Key]) -> [Value?] {

        var result = [Value?]()

        result.reserveCapacity(keys.count)

        for key in keys{

            result.append(self[key])

        }

        return result

    }

}

Swift中更简便的方法

小伙伴们可能会问,为什么Swift中不需要实现这么一个API呢?其实其有更简单的实现,如下面代码所示:


extension Dictionary {

    func valuesForKeys(keys: [Key]) -> [Value?] {

        return keys.map { self[$0] }

    }

}


上述方式实现的功能和最开始的方法实现的功能相同,虽然核心的功能是封装了map的调用,这个例子也说明了为什么Swift没有提供轻量级的API接口,因为小伙伴们简单的调用map就可以实现。

接下来,我们实验几个例子:


var dic: Dictionary = [ "1": 2, "3":3, "4":5 ]

var t = dic.valuesForKeys(["1", "4"]) //结果为:[Optional(2), Optional(5)]

var t = dict.valuesForKeys(["3", "9"]) // 结果为:[Optional(3), nil]

t = dic.valuesForKeys([]) //结果为:[]

内嵌可选类型

现在,如果我们为每一个结果调用last方法,看下结果如何?


var dic: Dictionary = [ "1": 2, "3":3, "4":5 ]

var t = dic.valuesForKeys(["1", "4"]).last //结果为:Optional(Optional(5)) // Optional(Optional("Ching"))

var t = dict.valuesForKeys(["3", "9"]).last // 结果为:Optional(nil)

var t = dict.valuesForKeys([]).last // 结果为:nil

小伙伴们立马迷糊了,为什么会出现两层包含的可选类型呢?,特别对第二种情况的Optional(nil),这是什么节奏?

我们回过头看看last属性的定义:


var last:T? { get }


很明显last属性的类型是数组元素类型的可选类型,这种情况下,因为元素类型是(String?),那么再结合返回的类型,于是其结果就是String??了,这就是所谓的嵌套可选类型。但嵌套可选类型本质是什么意思呢?
如果在Objective-C中重新调用上述方法,我们将使用NSNull作为占位符,Objective-C的调用语法如下所示:

[dict valuesForKeys:@[@"1", @"4"] notFoundMarker:[NSNull null]].lastObject

// 5

[dict valuesForKeys:@[@"1", @"3"] notFoundMarker:[NSNull null]].lastObject

// NSNull

[dict valuesForKeys:@[] notFoundMarker:[NSNull null]].lastObject

// nil


不管是Swift版本还是Objective-C版本,返回值为nil都意味数组是空的,所以它就没有最后一个元素。 但是如果返回是Optional(nil)或者Objective-C中的NSNull都表示数组中的最后一个元素存在,但是元素的内容是空的。在Objective-C中只能借助NSNull作为占位符来达到这个目的,但是Swift却可以语言系统类型的角度的实现。
提供一个默认值
进一步封装,如果我字典中的某个或某些元素不存在,我们想提供一个默认值怎么办呢?实现方法很简单:

extension Dictionary {

    func valuesForKeys( keys:[Key], notFoundMarker: Value)->[Value]{

        return self.valueForKeys(kes).map{ $0 ?? notFoundMarker }

    }

}

dict.valuesForKeys(["1", "5"], notFoundMarker: "Anonymous")


和Objective-C相比,其需要占位符来达到占位的目的,但是Swift却已经从语言类型系统的层面原生的支持了这种用法,同时提供了丰富的语法功能。这就是Swift可选类型的强大之处。同时注意上述例子中用到了空合运算符??。

 类似资料:
  • 本页包含内容: 为Dictionary增加objectsForKeys函数 Swift中更简便的方法 内嵌可选类型 提供一个默认值 可选类型是Swift中新引入的,功能很强大。在这篇博文里讨论的,是在Swift里,如何通过可选类型来保证强类型的安全性。作为例子,我们来创建一个Objective-C API的Swift版本,但实际上Swift本身并不需要这样的API。 为Dictionary增加ob

  • 本文向大家介绍swift 可选型的使用详解,包括了swift 可选型的使用详解的使用技巧和注意事项,需要的朋友参考一下  一、基本用法 可选性是Swift提供的一个特殊类型,它为我们编写程序提供便利的条件 swift是强类型语言,当我们需要使用一个变量,既可以为String,也可以为nil时,这时候就需要我们使用可选型。 二、可选型解包使用 三、多层解包(**swift3.0有变化) 四、Opti

  • 本文向大家介绍完美解决iview 的select下拉框选项错位的问题,包括了完美解决iview 的select下拉框选项错位的问题的使用技巧和注意事项,需要的朋友参考一下 在使用iview的过程中,我遇到这样一个问题,在Model中使用select下拉框组件。但是当弹出框超过一屏需要滚动时,select的下拉选项会出现错位(下图1为正常,图2为滚动后,下拉选项错位。) 图1: 图2: 在分析组件代

  • 本文向大家介绍vue.js中使用微信扫一扫解决invalid signature问题(完美解决),包括了vue.js中使用微信扫一扫解决invalid signature问题(完美解决)的使用技巧和注意事项,需要的朋友参考一下 1、点击按钮,实现微信扫一扫功能: 2、使用config接口注入配置信息,wx.config调用方法如下: (其中appId,timestamp,nonceStr,sign

  • Swift 的可选(Optional)类型,用于处理值缺失的情况。可选表示"那儿有一个值,并且它等于 x "或者"那儿没有值"。 Swfit语言定义后缀?作为命名类型Optional的简写,换句话说,以下两种声明是相等的: var optionalInteger: Int? var optionalInteger: Optional<Int> 在这两种情况下,变量 optionalIntege

  • 本文向大家介绍完美解决ARIMA模型中plot_acf画不出图的问题,包括了完美解决ARIMA模型中plot_acf画不出图的问题的使用技巧和注意事项,需要的朋友参考一下 问题描述:在画时间序列ACF时,调用 画不出图,或者是只能画出一条直线,如下图所示: 出现这种情况的原因是:plot_acf(data, lags=40)中的data没有dropna()。 解决方案: 结果如下: 补充知识:Py