swift中,我们经常看到和用到?和!,今天来聊一聊它们俩。
swift编程,不外乎是定义属性或者函数(方法),访问属性或者调用函数,类型转换,?和!在这几个过程中,都有一展身手的时候,而且,每次要考虑使用的时候,它们俩都会一起出现在我们的大脑中,用还是不用,如果用,要用谁?
1、定义属性
“?”表示可选类型(Optionals),“!”表示隐式可选类型(全名:implicitly unwrapped optionals),其实还是可选类型。
可选类型,就是将已存在的某种类型(结构体、枚举、类)定义为可选类型,表示该“新”类型的值可以为空nil。书写格式就是在原来的类型后面跟一个“?”比如:
var nickName : String?
定义了一个可选类型String?的变量nickName,如果我们不在构造函数(init这类函数)中给nickName赋初值,那么,系统会默认给它一个nil为初值,其实,我们定义的可选类型的时候,系统马上就把可选类型变量或常量初始化为nil了,在调用init方法之前。
在访问可选类型属性时,如果我们确定该属性在这个时候一定有值,可以在属性名称后面加“!”,告诉系统,“我肯定这个可选属性有值,强制取出来用”,但是,如果这个可选类型属性当时的值为nil,那么会crash。呵呵。(如果访问属性或者调用函数,必须用“!”或者“?”,下文再说),照样访问,这样,如果是nil,也不会因为这个导致crash。所以,一般在程序的关键点使用可选类型之前,都会做安全检查,判断是否为实际值是否为nil。判断方法有两种:
swift:
if nickName != nil { print("(nickName)") }
if let tempName = nickName { print("(tempName)") }
//第二种写法叫可选绑定,意思是,如果可选类型nickName不为nil,就取出其中的值,赋给tempName,并且类型推断tempName为NSString,因为确定等号后面的那个量不为nil才执行赋值,所以,tempName 不是可选类型,赋值成功,此时相当于if true {。。。},如果nickName此时的值为nil时的情况,请自行脑补。
“ !” ,隐式可选类型,就是可选类型,书写位置同 “ ?”,区别是,用“!”声明的可选类型,访问属性的时候都可以不用写“!”来取出属性的真实值,这就告诉系统,我知道这个属性在其存活过程中一直会有值。其实,在控制器中访问自己的可选类型属性是需要用“!”来取值的,这个动作官方叫强制解包(forced unwrapping)。上面说取值,为了方便理解。还能专业点么?下统一用解包。
var nickName :String!
注意:如果隐式可选类型属性在生命周期中可能会为nil,那还是不要将其定义为“!”类型的了。会crash的。
2、定义函数?(你在逗我?)
“?”和“!”是可以用来定义函数的。类、结构体、枚举都是需要构造器的,那么,有时候根据某种业务要求,我们会判定,本次实例构造过程失败,于是,就有了可失败构造器。官方文档里面举例如下:
class Product {
let name: String!
init?(name: String) {
self.name = name
if name.isEmpty { return nil }
}
}
所以,看到init?和init!的时候不要慌,就是要实例化的类型可能是空,可能构造(创建)失败,失败的时候会返回nil,虽然swift不想OC中那样需要在init方法中返回self。上例中,如果构造成功,比如:
let newProduct = Product("巧克力")
根据Product类定义,构造实例成功,newProduct的类型推断为Product?。(除非你硬要在接收实例的变量或常量后面加个可选类型,不要作,真的。。。)
作为基友,怎么能没有“!”的事!其实作用一样,只是,如果构造成功,返回实例的类型推断为Product!(隐式可选)
有一点需要强调的是,如果当前实例化的类为某个类的子子子类,比如我们自定义了一个UIButton的子类BaseButton,又定义了PYButton :BaseButton,实例化PYButton的时候,如果失败了,那么在实例化过程中继承者链条无论进行到哪一步了,就马上停止。
3、访问属性和调用函数
这俩一起聊了,在我们访问属性或者调用函数的时候,因为swift允许各种连语法连在一起用,所以,有时候会出现一大串点语法连续调用,属性访问和函数调用相互交叉,在这个过程中,如果遇到可选类型怎么办?要写一堆“!”解包?答案是否定的。
swift提供了一个机制,叫“可空链式调用”,在一连串的点语法调用和访问中,只需要在调用者后面加一个“?”,表示,如果这一串点语法执行过程中,任何一步失败,整个点语法链条就会停止,并且作为一个整体返回nil,不会报错,不会crash。那么,问题来了,用“!”不行么?可以!但是,如果在点语法链条中,那一步返回值为nil,就会导致crash。既然我们没法永远保证某块代码返回值不为nil,或者属性为nil,或者失败,那么用“ ?”是极好的。
上代码:
class Person {
var telString :String?
}
let xiaoMing = Person()
print("(xiaoMing.telString!.intValue)") //crash,因为telString为nil,强制解包就crash
print("(xiaoMing.telString?.intValue)") //输出:nil
4、类型转换
大家都知道OC中有多态,某些情况下,我们会将子类实例赋值给父类指针,用到的时候,再强转回子类。swift中,这种情况,使用as? 和 as! 来做,如果要判断某个实例的类型 用 is。 A as? B的意思是,如果实例A是B类型或者是B类型的子类,就将A的类型转化成B类型,并返回转换后的实例(是可选类型B?),如果不是,表达式返回nil,程序继续运行。如果用as! ,说明我们肯定A是类型B或者B的子类实例,那么,强制转换,如果不是,那么会crash。比如,
class Teacher : Person {
var teachingID : String
init(teachingID : String) {
self.teachingID = teachingID
super.init()
}
}
class Student : Person {
var studentID : String
init(studentID : String) {
self.studentID = studentID
super.init()
}
}
let teacher = Teacher("12345")
let student = Student("987")
let classContact = [teacher,student]
//注意:此时classContact的类型是:[Person]
for person in classContact {
if let teacher = person as? Teacher { //code}
else if let student = person as? Student { // code}
}
as!的情况自行脑补。简单提一下,is,它就相当于 OC中的 isKindOfClass:方法。
另外,还有两个问号连在一起用的,其实也很简单,先看代码:
let name = “panyu”
let nickName = name ?? "hehe" //如果name为nil(当然本例不会为nil),那么表达式返回值为"??"之后的值,如果name有值,那么表达式返回name本身的值
一句话,?? 表达式就是三目运算符的简单版。