可选类型:
开发中经常遇到将一个变量初始化为0,或将一个对象初始化为nil的情况,但在Swift中,nil是一个特殊的类型,而Swift中是不允许不同类型之间的相互赋值的,为了解决这个问题,Swift中引入了可选类型
错误写法:
let view : UIView = nil //UIView * view = nil;这种思路在这里可不适用哦
let name : String = nil //NSString * name = nil;
- 定义可选类型:
可选类型是一个泛型集合,必须告诉该可选类型中存放什么元素,它的标准定义方法如下:
var 变量名称 : Optional<泛型> = nil //Optional这个关键字表示可选类型,泛型用于约束这个变量可以选择的类型
Tips:
这句话的意思很简单:我定义的这个变量的类型可以选择为nil,也可以选择为泛型
当然,Optional后面的泛型只能写一个,超过一个则编译报错
例:
var name : Optional<String> = nil //完整写法,不常见
然而标准写法在Swift这门追究极简的语言中并不常见,真正常见的是下面这种甜甜的"语法糖"
例:
var name : String? = nil //语法糖,与上面例子的语义完全相同,常见
- 可选类型赋值
var name : String?
name = Optional("lyu") //赋值一个String?类型
name = "lyu" //赋值一个String类型
- 可选类型的取值
- 强制解包
print(name!) //强制解包
注意:如果可选类型中的值为nil,那么强制解包会导致程序崩溃,所以强制解包是一个危险的操作
- 可选绑定:
为了避免强制解包引起崩溃,我们需要在强制解包前判断一下被解包的变量是否真的有值,如果没有值就不对他进行解包,思路如下:
可选绑定的思路
var name : String?
name = "lyu"
if name != nil {
let tempName = name
print(tempName)
}
可选绑定:
var name : String?
name = "lyu"
if let name = name { //这里,两个name的作用域不同,实际上是在"内部"使用"外部"的变量给新创建的变量进行初始化
print(name)
}
解释:系统会先判断name是否有值,如果没有值就跳过大括号,如果有值,那么系统会对name进行强制解包,并将解包后的结果赋值给新创建的"局部变量"name
建议:初学者不要进行强制解包,避免引起崩溃6
- 可选类型的扩展
在Swift中,思考什么时候使用可选类型并不讨厌,而是不知不觉中我的某一个变量就变成了可选类型,这一点总是搞得新手不能愉快的编写代码~
然而并不可怕,我们可以发现按照Swift一贯的尿性,一个函数的返回值如果可能为空(nil),那么这个函数的返回值多半是一个可选类型
例1:实例化
var url = NSURL(string: "http://www.lyu.com") //实际上url已经是一个可选类型了,因为后面的方法有可能获取不到url,也就导致url有可能为空,所以Swift判定url为可选类型而不是NSUrl类型,其等同于下面一行代码:
var url : NSURL? = NSURL(string: "http://www.lyu.com")
//错误写法:
var url : NSURL = NSURL(string: "http://www.lyu.com") //此处编译报错,因为变量类型"NSURL"与返回值类型"Optional"不匹配
let request = NSURLRequest(URL: url!) //此处需要解包
例2:取文件
let infoPlist = NSBundle.mainBundle().pathForResource("infoPlist.plist", ofType: nil)
if let infoPlist = infoPlist { //如果mainBundle中真的有infoPlist.plist这个文件,那么就进行转字典操作,否则跳过
let infoDic = NSDictionary(contentsOfFile: infoPlist)
print(infoDic)
print(infoPlist)
}
例3:数据类型转换
let numberString = "1234"
let number : Int? = Int(numberString)
网友评论