Optional的定义

Optional也是Objective-C没有的数据类型,是苹果引入到Swift语言中的全新类型,它的特点就和它的名字一样:可以有值,也可以没有值,当它没有值时,就是nil。此外,Swift的nil也和Objective-C有些不一样,在Objective-C中,只有对象才能为nil,而在Swift里,当基础类型(整形、浮点、布尔等)没有值时,也是nil,而不是一个初始值,没有初始值的值,是不能使用的,这就产生了Optional类型。定义一个Optional的值很容易,只需要在类型后面加上问号(?)就行了,如:

var str: String?

一个Optional值和非Optional值的区别就在于:Optional值未经初始化虽然为nil,但普通变量连nil都没有:

//未被初始化,但是是一个Optional类型,为nil

var str: String?

str //输出nil

//未被初始化,也不是Optional类型

var str2: String

str2    //使用时出错

Optional的拆包

显式拆包

Optional类型的值不能被直接使用,当需要用时要显式拆包,以表明我知道这个Optional是一定有值的:

var str: String? = "Hello World!"

str! //Hello World!

对比拆包前后,对str的输出:

var str: String? = "Hello World!"

str     //{Some "Hello World!"}

str!    //Hello World!

之所以要拆包使用,是因为Optional类型其实是一个枚举:

enum Optional<T> : Reflectable, NilLiteralConvertible {

case None

case Some(T)

init()

init(_ some: T)

/// Haskell's fmap, which was mis-named

func map<U>(f: (T) -> U) -> U?

func getMirror() -> MirrorType

static func convertFromNilLiteral() -> T?

}

当Optional没有值时,返回的nil其实就是Optional.None,即没有值。除了None以外,还有一个Some,当有值时就是被Some<T>包装的真正的值,所以我们拆包的动作其实就是将Some里面的值取出来。

有没有似曾相识的感觉?Java里面也有泛型。

隐式拆包

除了显式拆包,Optional还提供了隐式拆包,通过在声明时的数据类型后面加一个感叹号(!)来实现:

var str: String! = "Hello World!"

str //Hello World!

可以看到没有使用(?)进行显式的折包也得到了Some中的值,这个语法相当于告诉编译器:在我们使用Optional值前,这个Optional值就会被初始化,并且总是会有值,所以当我们使用时,编译器就帮我做了一次拆包。如果你确信你的变量能保证被正确初始化,那就可以这么做,否则还是不要尝试为好。

另外:在上面可以看到,Optional其实就是一个枚举,然后给它指定一个类型就行了,所以下面这两种方法都能声明一个Optional值:

var str: String! = "Hello World!"

var str2: Optional<String>

Optional Binding

在说Optional Binding之前,我想先说下Xcode6 Beta5在这一版中的一个小变化:在Xcode6 Beta5之前,如果是一个Optional值,可以直接放到条件判断语句中,如:

var str: String? = "Hello World!"

if str {

"not nil"

} else {

"nil"

}

如果不是nil,则右边的Playground会显示“not nil”;反之则显示“nil”,但是至Xcode6 Beta5开始,这样就不能通过编译器了,你需要用下面这种方式来代替:

var str: String? = "Hello World!"

if str != nil {

"not nil"

} else {

"nil"

}

看似合理,但是在某种情况下会非常不爽,比如你在str != nil条件成真后接着在上下文中使用str,会被要求进行拆包,我们以一个Int类型的Optional来做示例:

var count: Int?

count

if count != nil {

"count is " + String(count!) 
  //count is 100

} else {

"nil"

}

我在把count强转成String的时候被要求拆包了,这是因为count本身是一个Optional的类型,为了避免在条件判断语句后执行一次或更多次的拆包,Swift引进了Optional Binding,我们就可以这样做:

var count: Int?

count

if let validCount = count {

"count is " + String(validCount)    //count is 100

} else {

"nil"

}

通过在条件判断语句中(如if、while等)把Optional值直接给一个临时常量,Swift会自动检测Optional是否包含值,如果包含值,会隐式的拆包并给那个临时常量,在接下来的上下文中就能直接使用这个临时常量了,这样是不是就觉得很爽呢

注:在Optional Binding中,除了以常量的方式去接收拆包的值之外,也能以一个变量的形式

去接收,但相信在大多数情况下我们只是使用那个值就行了,并不会去改变它。

Optional Chaining

Optional Chaining对Swift来说是很基本但又必不可少的东西,相对于简单类型(Int、String等)来说,Optional更主要的应用场景是在复杂对象上,当一个对象包含另一个对象,同时这两个对象都有可能为nil的情况下才是Optional派上用场的地方,在Objective-C里,向nil发消息得到的就是一个nil,但是Swift不能在nil上直接调用方法或属性,同时为了方便我们使用,从而引入了Optional类型,可是这还不够,我们做一个简单的例子:

class Person {

var pet: Pet?

}

class Pet {

var name: String

var favoriteToy: Toy?

init (name: String) {

self.name = name

}

}

class Toy {

var name: String

init (name: String) {

self.name = name

}

}

一个Person对象代表一个人,这个人可能有一个宠物,宠物会有它自己的名字,而且宠物可能会有自己喜爱的玩具,按照前面提到的知识,我们要首先判断这个人有没有宠物,然后再判断他的宠物有没有喜爱的玩具,然后才能得到这个玩具的名称,利用Optional Binding,我们写出来的可能就像这样:

let jackon = Person()

jackon.pet = Pet(name: "Max")

jackon.pet?.favoriteToy = Toy(name: "Ball")

if let pet = jackon.pet {

if let toy = pet.favoriteToy {

toy.name

}

}

这里用到了两个if,因为pet和toy对象都可能为nil,我们需要预防每一个可能为nil的对象,如果这个对象再复杂一点,那if也就更多了,而使用Optional Chaining的话,写出来的就像这样:

let jackon = Person()

jackon.pet = Pet(name: "Max")

jackon.pet?.favoriteToy = Toy(name: "Ball")

if let toy = jackon.pet?.favoriteToy {

toy.name

}

当一个Optional值调用它的另一个Optional值的时候,Optional Chaining就形成了,基本上,Optional Chaining就是总是返回一个Optional的值,只要这个Chaining中有一个值为nil,整条Chaining就为nil,和Objective-C的向nil发消息类似。

有一点很有趣,就是Optional Chaining除了能将属性返回的类型变为Optional外,连方法的返回值都能强制变为Optional,哪怕这个方法没有返回值,但是别忘了,Void也算是一个类型:

typealias Void = ()

如果我们的Pet类有一个玩玩具的play方法的话,就可以这样来判断是否会调用成功:

if let p: Void = jackon.pet?.play()
{

"play is called"

}

使用Optional Chaining,能使我们的代码变得更加可读,同时更加简洁。

转自:http://blog.csdn.net/zhangao0086/article/details/38640209

Swift 可选值(Optional Values)介绍的更多相关文章

  1. swift的可选值(optional)

    苹果那文档写了一大堆也没有好好的写一下可选值(optional)这个东西.就是在有一个“Optional Chaining”的章节,但是也不是很充分的说明.最后找了半天在“the basics”里墨迹 ...

  2. swift可选值总结

    1.枚举结构: 2.装包.解包概念:关联值. 3.可选值声明: 4.解包: 5.可选值作为参量生成的链. 6.可选值调用链. 最后做个总结 访问可选对象的属性或方法时,可以用 ? 号 访问可选对象的属 ...

  3. Swift可选类型(Optional)之星耀

    首先我们先看下Objective-C与Swift语言对于可选nil的不同理解: Objective-C中的nil:表示缺少一个合法的对象,是指向不存在对象的指针,对结构体.枚举等类型不起作用(会返回N ...

  4. 2. Swift元组|可选值|断言

    1. 元组英文名字 Tuple,将多个数据类型(任意类型)组合成一个数据,与c语言的中的机构体有几分相似,功能也是非常强大的,尤其是在定义请求参数,状态之类的地方经常用到. let http404Er ...

  5. Swift学习笔记(2)--元组(Tuples)、Optional(可选值)、(Assertions)断言

    1.Tuples(元组) 元组是多个值组合而成的复合值.元组中的值可以是任意类型,而且每一个元素的类型可以是不同的. 1>定义:使用()包含所有元素,用逗号分开,也可以对每个元素做命名 let ...

  6. 窥探Swift之新添数据类型元组与可选值

    今天的博客中就总结一下关于Swift中相对Objc新添加的两个数据类型:元组(Tuple)和可选值类型(Optional).上面这两个类型是Swift独有的类型,使用起来也是非常方便的,今天就通过一些 ...

  7. Swift中的Optional类型 (可选类型)与强制解包 ? !

    我们在swift的开发中会经常遇见?和! ,理解这两个符号深层次的内容对我们的开发是相当有利的: 目前网上对swift3.0的教程还相当的少,如果去搜索会发现早期的说法,在定义变量的时候,swift是 ...

  8. Swift语言指南(六)--可选值

    原文:Swift语言指南(六)--可选值 在值可能不存在的情况下使用可选值(optional), 可选值是: · 存在一个值,这个值等于 x 或 · 不存在任何值 注: 在 C 和 Objective ...

  9. swift 可选类型(optional)

    可选类型定义 Swift 标准库中定义后缀  ?为可选类型 Optional<Wrapped> 的语法糖,这里语法糖可以简单理解为一种便捷的书写语法.也就是说,下面两个声明是等价的: va ...

随机推荐

  1. 通过外网IP访问内网

    外网服务器:外网IP1,内网IP192.168.2.156 内网服务器:内网IP192.168.2.206 通过访问外网服务器8083端口,转发到内网服务器的8083端口. 在外网服务器设置映射规则: ...

  2. UVa 10935 (水题) Throwing cards away I

    直接用STL里的queue模拟即可. #include <cstdio> #include <queue> using namespace std; ; int discard ...

  3. PHP后台执行

    php中实现后台执行的方法: ignore_user_abort(true); // 后台运行set_time_limit(0); // 取消脚本运行时间的超时上限后台运行的后面还要,set_time ...

  4. Linux ARM kernel Makefile and Kconfig

    kernel build:顶层Makefile:-->1. include build/main.mk    -->2. include build/kernel.mk         k ...

  5. IOS中bounds和frame

    * 用bounds和frame来修改尺寸是有一些小区别的 三.isEqual:方法 1> 系统会根据对象isEqual方法的返回值来决定两个对象是否相同 * 比如判断对象a和b是否相同,就会查看 ...

  6. HDU2028 Lowest Common Multiple Plus

    解题思路:最近很忙,有点乱,感觉对不起自己的中国好队友.   好好调整,一切都不是问题,Just do it ! 代码: #include<cstdio> int gcd(int a, i ...

  7. LCS nlog(n) 但最坏情况还是比较悲剧 转载的文章;

    最长公共子序列问题: 给定2个字符串,求其最长公共子串.如abcde和dbada的最长公共字串为bd. 动态规划:dp[i][j]表示A串前i个和B串前j个的最长公共子串的长度. 则 若A[i] == ...

  8. MAC OSX 下安装Cscope

    续前文,搞定CTAGS之后,需要被搞定的是cscope,依旧是上网拖一把,具体过程如下   #1 下载cscope最新版本 http://cscope.sourceforge.net/#downloa ...

  9. 写python时加入缩进设置

    发现如果用vim写python的时候,还是设成8好像会报错,在现有的基础上,加入下面设置就好了set shiftwidth=4

  10. AE+C# 向axPageLayoutControl1添加图例

    原文 AE+C# 向axPageLayoutControl1添加图例 //Get the GraphicsContainer IGraphicsContainer graphicsContainer ...