使用self.amiteLbl!
是明确声明此时self.amiteLbl
里一定有值,无论什么情况都调用后面的.text赋值方法。
而使用self.view?
是声明此时并不确定self.view里是否有值,所以只在view有值时调用后面.addSubview
方法。
这样做的目的一是让代码更明确, 二是给编译器提供更多线索,在编译时发现更多潜在错误。
声明为Optional只需要在类型后面紧跟一个?
即可。如:
var strValue: String? //?相当于下面这种写法的语法糖
var strValue: Optional<String>
上面这个Optional的声明,意思不是”我声明了一个Optional的String值”, 而是”我声明了一个Optional类型值,它可能包含一个String值,也可能什么都不包含”,也就是说实际上我们声明的是Optional类型,而不是声明了一个String类型,这一点需要铭记在心。
然后怎么使用Optional值呢?文档中也有提到说,在使用Optional值的时候需要在具体的操作,比如调用方法、属性、下标索引等前面需要加上一个?
,如果是nil值,也就是Optional.None
,会跳过后面的操作不执行,如果有值,就是Optional.Some
,可能就会拆包(unwrap),然后对拆包后的值执行后面的操作,来保证执行这个操作的安全性,比如:
let hashValue = strValue?.hashValue
strValue是Optional的字符串,如果strValue是nil,则hashValue也为nil,如果strValue不为nil,hashValue就是strValue字符串的哈希值(其实也是用Optional wrap后的值)
另外,?还可以用在安全地调用protocol类型方法上,比如:
@objc protocol Downloadable {
@optional func download(toPath: String) -> Bool;
}
@objc class Content: Downloadable {
//download method not be implemented
}
var delegate: Downloadable = Downloadable()
delegate.download?("some path")
因为上面的delegate是Downloadable类型的,它的download
方法是optional,所以它的具体实现有没有download
方法是不确定的。Swift提供了一种在参数括号前加上一个?
的方式来安全地调用protocol的optional方法。
另外如果你需要像下面这样向下转型(Downcast),可能会用到 as?
:
if let dataSource = object as? UITableViewDataSource {
let rowsInFirstSection = dataSource.tableView(tableView, numberOfRowsInSection: 0)
}
到这里我们看到了
?
的几种使用场景:
1.声明Optional值变量
2.用在对Optional值操作中,用来判断是否能响应后面的操作
3.用于安全调用protocol的optional方法
4.使用 as? 向下转型(Downcast)
直接上例子,strValue是Optional的String:
let hashValue = strValue!.hashValue
这里的
!
表示“我确定这里的的strValue一定是非nil的,尽情调用吧” ,比如这种情况:
if strValue {
let hashValue = strValue!.hashValue
}
考虑下这一种情况,我们有一个自定义的
MyViewController
类,类中有一个属性是
myLabel
,myLabel是在viewDidLoad中进行初始化。因为是在viewDidLoad中初始化,所以不能直接声明为普通值:
var myLabel : UILabel
,因为非Optional的变量必须在声明时或者构造器中进行初始化,但我们是想在viewDidLoad中初始化,所以就只能声明为Optional:
var myLabel: UILabel?
, 虽然我们确定在viewDidLoad中会初始化,并且在ViewController的生命周期内不会置为nil,但是在对myLabel操作时,每次依然要加上
!
来强制拆包(在读取值的时候,也可以用
?
,谢谢iPresent在回复中提醒),比如:
myLabel!.text = "text"
myLabel!.frame = CGRectMake(0, 0, 10, 10)
...
对于这种类型的值,我们可以直接这么声明:var myLabel: UILabel!
, 果然是高(hao)大(gui)上(yi)的语法!, 这种是特殊的Optional,称为Implicitly Unwrapped Optionals
, 直译就是隐式拆包的Optional,就等于说你每次对这种类型的值操作时,都会自动在操作前补上一个!
进行拆包,然后在执行后面的操作,当然如果该值是nil,也一样会报错crash掉。
var myLabel: UILabel! //!相当于下面这种写法的语法糖
var myLabel: ImplicitlyUnwrappedOptional<UILabel>
那么
!
大概也有两种使用场景
1.强制对Optional值进行拆包(unwrap)
2.声明Implicitly Unwrapped Optionals
值,一般用于类中的属性
http://joeyio.com/ios/2014/06/04/swift---/