我对这个的理解就是:我们有可能会用到其他的属性或者方法,当我们在使用其他的时候,可以使用点语法去访问另一个的属性,这样的使用,就形成了链式访问。
可空链式调用是一种可以请求和调用属性、方法及下表的过程,它的可空性体现于请求调用的目标当前可能为空(nil)。如果可空的目标有值,那么调用就会成功;如果选择的目标为空nil,那么这个调用将返回空。多个连续的调用可以被链接在一起形成一个调用链,如果其中任何一个节点为空nil将导致整个链调用失败。
Attention:Swift的可空链式调用和OC中的消息为空有些相像,但是Swift可以使用在任意类型中,并且能够检查调用是否为空。
------如何定义为可空链呢?
在属性、方法、下表的可空值后面放一个问号,可以定义一个可空链。这一点很像在可空值后面放一个叹号!来强制展开其中值。他们的区别是,当可空值为空时,可空链式只是调用失败,然而强制展开将会触发运行时错误。
如果调用一个空对象的任何属性,将返回可空值,无论它的属性是不是可空值。可以通过返回值来判断你的可空链式调用是否调用成功,如果有返回值,则调用成功;如果返回nil则调用失败。
Expecially:可空链式调用的返回结果与原本的返回结果具有相同的类型,但是被包装成了一个可空类型值。当可空链式调用成功时,一个本应该返回Int的类型的结果将会返回Int?类型。
可空类型长什么样子呢?示例如下:
class Person { var residence :Residence? //这个问号表明了residence是一个可空的属性,它的类型为Residence?
}
如果新建一个Person实例,因为它的residence属性是可空的,john的属性将初始化为nil。
let john = Person() let roomCount = john.residence!.numberOfRooms 报错信息:fatal error: unexpectedly found nil while unwrapping an Optional value
当使用叹号!强制展开获得这个john的residence属性中的numberOfRooms值,会触发运行时错误,因为没有可以展开的residence。。
class Person { var residence :Residence? } class Residence { var numberOfRooms = 1 } let john = Person() let res:Residence = Residence() let roomCount = john.residence?.numberOfRooms print(roomCount)
print:nil
因为访问numberOfRooms有可能失败,可空链式调用会返回Int?类型,或者称为“可空的Int”。当实例为nil,将返回nil。
当给person的residence赋值后,就可以用返回值了:
let john = Person() john.residence = Residence() let roomCount = john.residence?.numberOfRooms print(roomCount) 打印结果:Optional(1)
****为可空链式调用定义模型类
通过使用可空链式调用可以调用多层属性,方法,和下标。这样就可以通过各种模型向下访问各种子属性,并且判断能否访问子属性的属性、方法或下标。
-----通过可空链式调用来访问下标
可以通过下标来对可空值进行读取或写入,并且判断下标调用是否成功。当通过可空链式调用访问可空值的下标的时候,应该将问号放在下标方括号的前面,可空链式调用的问号一般直接跟在可空表达式的后面。
let johsnHouse = Residence() johsnHouse.rooms.append(Room(name: "Living Room")) johsnHouse.rooms.append(Room(name: "Kitchen")) john.residence = johsnHouse if let roomCount = john.residence?.numberOfRooms{ print(roomCount) }else{ print("ni;") } if let firstRoomName = john.residence?[0].name { print(firstRoomName) }else{ print("meiyou") } 运行结果: 2 Living Room
-----访问可空类型的下标
如果下标返回可空类型值,比如在Swift中Dictionary的key下标,可以在下标的闭合括号后面放一个问号来链接下标的可空返回值。
var testScores = ["Bev":[79,94,81],"Dave":[86,82,84]] testScores["Dave"]?[0] = 91 testScores["Bev"]?[0]++ testScores["Brian"]?[0] = 72--->调用失败 print(testScores) 打印: ["Bev": [80, 94, 81], "Dave": [91, 82, 84]]
-----多层链接
可以通过链接多个可空链式调用来向下访问属性、方法及下标。但是多层可空链式调用不会添加返回值的可空性。
1.如果你访问的值不是可空的,通过可空链式调用将会返回可空值;
2.如果你访问的值已经是可空的,通过可空链式调用不会变的“更”可空;
if let johnStreet = john.residence?.address?.street { print(johnStreet) }else{ print("not street") }
let someAddress = Address() someAddress.buildingNumber = "The 29" someAddress.buildingName = "The ABC" john.residence?.address = someAddress if let beginsWith = john.residence?.address?.buildingIdentifier()?.hasPrefix("The"){ if beginsWith{ print("the beig") }else{ print("not the") } }else{ print(john.residence?.address?.buildingIdentifier()) }
在方法的圆括号后面加上问号,是因为buildIdentifier()的返回值是可空值,而不是方法本身是可空的。