From b42ceb0e5c891ff88a9fe72f619fbf77ecb29c20 Mon Sep 17 00:00:00 2001 From: littlekok Date: Fri, 29 Aug 2014 19:49:13 +0800 Subject: [PATCH] =?UTF-8?q?=E6=A0=A1=E6=AD=A3=E4=BA=86=E9=83=A8=E5=88=86?= =?UTF-8?q?=E6=A0=87=E7=82=B9=E3=80=81=E8=AF=AD=E5=8F=A5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- source/chapter4/07_Optional_Case_Study.md | 39 ++++++++++++++++------- 1 file changed, 27 insertions(+), 12 deletions(-) diff --git a/source/chapter4/07_Optional_Case_Study.md b/source/chapter4/07_Optional_Case_Study.md index 037b7f2d..012e0806 100644 --- a/source/chapter4/07_Optional_Case_Study.md +++ b/source/chapter4/07_Optional_Case_Study.md @@ -11,25 +11,25 @@ - [内嵌可选类型](#nested-optional) - [提供一个默认值](#provide-default) -可选类型是Swift中新引入的,功能很强大,这篇博文我们将讨论在Swift里如何通过可选类型来保证强类型的安全性。我们创建一个Objective-C API的Swift版本作为例子,但实际上Swift本身并不需要这样的API。 +可选类型是Swift中新引入的,功能很强大。在这篇博文里讨论的,是在Swift里,如何通过可选类型来保证强类型的安全性。作为例子,我们来创建一个Objective-C API的Swift版本,但实际上Swift本身并不需要这样的API。 #### 为Dictionary增加objectsForKeys函数 -在Objective-C中,```NSDictionary```有一个方法```-objectsForKeys:NoFoundMarker:```, 这个方法需要一个键值值的```NSArray```数组作为参数,同时返回一个包含相关值的数组。从文档里我们可以看出:"返回数组中的第N个值和参数输入数组中的第N个值相对应",那如果有某个键值在字典里不存在呢?于是就以```notFoundMarker```作为返回提示,比如第三个键值没有找到,那么在返回数组中第三个值就是```notFoundMarker```,而不是字典中的第三个值,但是这个值只是提醒原字典中没有找到,但在返回数组中该元素存在,且用```notFoundMarker```作为占位符,因为其不能直接使用,所以在Foundation框架中有个专门的类处理这个情况:```NSNull```。 +在Objective-C中,```NSDictionary```有一个方法```-objectsForKeys:NoFoundMarker:```, 这个方法需要一个```NSArray```数组作为键值参数,然后返回一个包含相关值的数组。文档里写到:"返回数组中的第N个值,和输入数组中的第N个值相对应",那如果有某个键值在字典里不存在呢?于是就有了```notFoundMarker```作为返回提示。比如第三个键值没有找到,那么在返回数组中第三个值就是这个```notFoundMarker```,而不是字典中的第三个值,但是这个值只是用来提醒原字典中没有找到对应值,但在返回数组中该元素存在,且用```notFoundMarker```作为占位符,因为这个对象不能直接使用,所以在Foundation框架中有个专门的类处理这个情况:```NSNull```。 -在Swift中,```Dictionary```类没有```objectsForKeys```的函数,为了说明问题,我们动手加一个,并且使其成为操作字典值的通用方法。我们可以使用```extension```来实现: +在Swift中,```Dictionary```类没有类似```objectsForKeys```的函数,为了说明问题,我们动手加一个,并且使其成为操作字典值的通用方法。我们可以用```extension```来实现: ``` extension Dictionary{ func valuesForKeys(keys:[K], notFoundMarker: V )->[V]{ - //接下来实现 + //具体实现代码后面会写到 } } ``` -以上就是我们实现的Swift版本,这个Objective-C版本有很大区别, 在Swift中,因为其强类型的原因限制了返回的结果数组只能包含单一类型的元素,所以我们不能放```NSNull```在字符串数组中,但是,Swift有更好的选择,我们可以返回一个可选类型数据,我们所有的值都封包在可选类型中,而不是```NSNull```, 我们只用```nil```即可。 +以上就是我们实现的Swift版本,这个和Objective-C版本有很大区别。在Swift中,因为其强类型的原因限制了返回的结果数组只能包含单一类型的元素,所以我们不能放```NSNull```在字符串数组中,但是,Swift有更好的选择,我们可以返回一个可选类型数据。我们所有的值都封包在可选类型中,而不是```NSNull```, 我们只用```nil```就可以了。 ``` @@ -48,7 +48,7 @@ extension Dictionary{ #### Swift中更简便的方法 -小伙伴们可能会问,为什么Swift中不需要实现这么一个API呢?其实其有更简单的实现,如下代码所示: +小伙伴们可能会问,为什么Swift中不需要实现这么一个API呢?其实其有更简单的实现,如下面代码所示: ``` extension Dictionary { @@ -58,14 +58,21 @@ extension Dictionary { } ``` -上述方式实现的功能和最开始的方法实现的功能相同,虽然核心的功能是封装了```map```的调用,这个例子也说明了为什么Swift没有提供轻量级的API接口,因为小伙伴们可以通过简单的调用```map```就可以实现。 +上述方式实现的功能和最开始的方法实现的功能相同,虽然核心的功能是封装了```map```的调用,这个例子也说明了为什么Swift没有提供轻量级的API接口,因为小伙伴们简单的调用```map```就可以实现。 接下来,我们实验几个例子: ``` var dic: Dictionary = [ "1": 2, "3":3, "4":5 ] -var t = dic.valuesForKeys(["1", "4"]) //结果为:[Optional(2), Optional(5)] -t = dic.valuesForKeys([])//结果为:[] + +var t = dic.valuesForKeys(["1", "4"]) +//结果为:[Optional(2), Optional(5)] + +var t = dict.valuesForKeys(["3", "9"]) +// 结果为:[Optional(3), nil] + +t = dic.valuesForKeys([]) +//结果为:[] ``` @@ -75,8 +82,16 @@ t = dic.valuesForKeys([])//结果为:[] ``` var dic: Dictionary = [ "1": 2, "3":3, "4":5 ] + var t = dic.valuesForKeys(["1", "4"]).last //结果为:Optional(Optional(5)) -t = dic.valuesForKeys([])//结果为:Optional(nil) +// Optional(Optional("Ching")) + +var t = dict.valuesForKeys(["3", "9"]).last +// 结果为:Optional(nil) + +var t = dict.valuesForKeys([]).last +// 结果为:nil + ``` 小伙伴们立马迷糊了,为什么会出现两层包含的可选类型呢?,特别对第二种情况的```Optional(nil)```,这是什么节奏? @@ -105,7 +120,7 @@ var last:T? { get } #### 提供一个默认值 -进一步封装,如果我字典中的某个或某些元素不存在,我们想提供一个默认值怎么办呢?这很简单实现: +进一步封装,如果我字典中的某个或某些元素不存在,我们想提供一个默认值怎么办呢?实现方法很简单: ``` extension Dictionary { @@ -122,6 +137,6 @@ dict.valuesForKeys(["1", "5"], notFoundMarker: "Anonymous") 和Objective-C相比,其需要占位符来达到占位的目的,但是Swift却已经从语言类型系统的层面原生的支持了这种用法,同时提供了丰富的语法功能。这就是Swift可选类型的强大之处。同时注意上述例子中用到了空合运算符```??```。 ----------------- -本章节不是老码的原创,老码认真的阅读了苹果的官方博客,且自己的练习总结,如果小伙伴们费了吃奶的劲还是看不懂,请找度娘谷歌,还是看不懂请到老码[官方微博](http://weibo.com/u/5241713117)咆哮。 +本章节不是老码的原创,是老码认真的阅读了苹果的官方博客,自己的练习总结,如果小伙伴们费了吃奶的劲还是看不懂,请找度娘谷歌。还是看不懂?请到老码[官方微博](http://weibo.com/u/5241713117)咆哮。 ##### 本文由翻译自Apple Swift Blog :[Optionals Case Study: valuesForKeys](https://developer.apple.com/swift/blog/?id=12)