类型转换
_类型转换_可以判断实例的类型,也可以将该实例在其所在的类层次中视为其父类或子类的实例。
Swift 中类型转换的实现为is 和as 操作符。这两个操作符使用了一种简单传神的方式来检查一个值的类型或将某个值转换为另一种类型。
如同协议实现的检查(此处应有链接)中描述的那样,你还可以使用类型转换来检查类型是否遵循某个协议。
为类型转换定义类层次
你可以在类及其子类层次中使用类型转换来判断特定类实例的类型并且在同一类层次中将该实例类型转换为另一个类。下面的三段代码定义了一个类层次以及一个包含了这些类实例的数组,作为类型转换的例子。
第一个代码片段定义了一个叫做MediaItem 的新基类。这个类为出现在数字媒体库中的所有成员提供了基本的功能。它声明了一个String 类型的name 和一个叫做init 的name 初始化器。(这里假设所有的媒体项目,包括所有电影和音乐,都有一个名字。)
class MediaItem {
var name: String
init(name: String) {
self.name = name
}
}
下一个片段定义了两个MediaItem 的子类。第一个子类,Movie ,封装了额外的电影的信息。他在MediaItem 的基础上添加了名为director 的属性及其初始化器。第二个子类,Song ,增加了名为artist 的属性及其初始化器。
class Movie: MediaItem {
var director: String
init(name: String, director: String) {
self.director = director
super.init(name: name)
}
}
class Song: MediaItem {
var artist: String
init(name: String, artist: String) {
self.artist = artist
super.init(name: name)
}
}
最后一个代码段创建了名为library 的常量数组,它包含了两个Movie 实例和三个Song 实例。library 数组的类型是在初始化时根据常量字面量推断出来的。 Swift 的类型检查器能够推断Movie 和Song 有一个共同的父类MediaItem ,因此library 的类型推断为[MediaItem] :
let library = [
Movie(name: "Casablanca", director: "Michael Curtiz"),
Song(name: "Blue Suede Shoes", artist: "Elvis Presley"),
Movie(name: "Citizen Kane", director: "Orson Welles"),
Song(name: "The One And Only", artist: "Chesney Hawkes"),
Song(name: "Never Gonna Give You Up", artist: "Rick Astley")
]
// "library" 的类型被推断为[MediaItem]
事实上library 储存的项目在后台依旧是Movie 和Song 实例。总之,如果你遍历这个数组的内容,你取出的项目将会是MediaItem 类型而非Movie 或Song 类型。为了使用他们原生的类型,你需要_检查_他们的类型或将他们_向下转换_为不同的类型,如下所述。
类型检查
使用_类型检查操作符_ (is )来检查一个实例是否属于一个特定的子类。如果实例是该子类类型,类型检查操作符返回true ,否则返回false 。
下面的例子定义了两个变量,movieCount 和songCount ,用来计算数组library 中Movie 和Song 实例的个数:
var movieCount = 0
var songCount = 0
for item in library {
if item is Movie {
movieCount += 1
} else if item is Song {
songCount += 1
}
}
print("Media library contains \(movieCount) movies and \(songCount) songs")
// Prints "Media library contains 2 movies and 3 songs"
这个例子遍历了library 数组中的每个元素。每一轮中,for-in 的循环都将item 常量设置为数组中的下一个MediaItem 。
如果当前MediaItem 是Movie 类型的实例,item is Movie 返回true ,反之返回false 。同样的,item is Song 检查了该对象是否为Song 类型的实例。在for-in 循环的最后,movieCount 和songCount 的值就是数组中对应类型实例的数量。
向下类型转换
某个类类型的常量或变量可能实际上在后台引用自一个子类的实例。当你遇到这种情况时你可以尝试使用_类型转换操作符_(as? 或as! )将它_向下类型转换_至其子类类型。
由于向下类型转换能失败,类型转换操作符就有了两个不同形式。条件形式,as? ,返回了一个你将要向下类型转换的值的可选项。强制形式,as! ,则将向下类型转换和强制展开结合为一个步骤。
如果你不确定你向下转换类型是否能够成功,请使用条件形式的类型转换操作符 (as? )。使用条件形式的类型转换操作符总是返回一个可选项,如果向下转换失败,可选值为nil 。这允许你检查向下类型转换是否成功。
当你确信向下转换类型会成功时,使用强制形式的类型转换操作符(as! )。当你向下转换至一个错误的类型时,强制形式的类型转换操作符会触发一个运行错误。
下面的例子遍历了library 中的每个MediaItem ,并打印出相应的描述信息。要这样做的话,每个项目均需要被当做Movie 或Song 来访问,而不仅仅是MediaItem 。为了在描述信息中访问Movie 或Song 的director 和artist 属性,这样做是必要的。
在这个例子中,数组中每一个项目的类型可能是Movie 也可能是Song 。你不知道遍历时项目的确切类型是什么,所以这时使用条件形式的类型转换符(as? )来检查遍历中每次向下类型转换:
for item in library {
if let movie = item as? Movie {
print("Movie: '\(movie.name)', dir. \(movie.director)")
} else if let song = item as? Song {
print("Song: '\(song.name)', by \(song.artist)")
}
}
// Movie: 'Casablanca', dir. Michael Curtiz
// Song: 'Blue Suede Shoes', by Elvis Presley
// Movie: 'Citizen Kane', dir. Orson Welles
// Song: 'The One And Only', by Chesney Hawkes
// Song: 'Never Gonna Give You Up', by Rick Astley
例子开头尝试将当前item 当做Movie 向下类型转换。由于item 是一个MediaItem 的实例,它_有可能_是Movie 类型;同样的,也有可能是Song 或者仅仅是MediaItem 基类。介于这种不确定性,类型转换符as? 在向下类型转换到子类时返回了一个可选项。item as? Movie 的结果是Movie? 类型,也就是“可选Movie 类型”。
当数组中的Song 实例使用向下转换至Movie 类型时会失败。为了处理这种情况,上面的例子使用了可选绑定来检查可选Movie 类型是否包含了一个值(或者说检查向下类型转换是否成功)。这个可选绑定写作“if let movie = item as? Movie ”,它可以被读作:
尝试以Movie 类型访问item 。如果成功,设置一个新的临时常量movie 储存返回的可选Movie 类型 。
如果向下类型转换成功,movie 的属性将用于输出Movie 实例的描述信息,包括director 的名字。同理,无论是否在数组中找到Song ,均可以检查Song 实例然后输出合适的描述(包括artist 的名字)。
注意
类型转换实际上不会改变实例及修改其值。实例不会改变;它只是将它当做要转换的类型来访问。
Any 和 AnyObject 的类型转换
Swift 为不确定的类型提供了两种特殊的类型别名: -AnyObject 可以表示任何类类型的实例。 -Any 可以表示任何类型,包括函数类型。
只有当你确切需要使用它们的功能和行为时再使用Any 和AnyObject 。在写代码时使用更加明确的类型表达总要好一些。
这里有一个使用Any 类型来对不同类型进行操作的例子,包含了函数类型以及非类类型。这个例子定义了一个名为things 的数组,它用于储存Any 类型的值:
var things = [Any]()
things.append(0)
things.append(0.0)
things.append(42)
things.append(3.14159)
things.append("hello")
things.append((3.0, 5.0))
things.append(Movie(name: "Ghostbusters", director: "Ivan Reitman"))
things.append({ (name: String) -> String in "Hello, \(name)" })
这个things 数组包含了两个Int 值、两个Double 值、一个String 值、一个(Double, Double) 的元组、Movie 实例“Ghostbusters”、以及一个接收String 值并返回String 值的闭包表达式。
你可以在switch 结构的 case 中使用is 和as 操作符找出已知Any 或AnyObject 类型的常量或变量的具体类型。下面的例子使用switch 语句遍历了things 数组并查询每一项的类型。其中几个switch 的case 将确定的值和确定类型的常量绑定在一起,使其值可以被输出:
for thing in things {
switch thing {
case 0 as Int:
print("zero as an Int")
case 0 as Double:
print("zero as a Double")
case let someInt as Int:
print("an integer value of \(someInt)")
case let someDouble as Double where someDouble > 0:
print("a positive double value of \(someDouble)")
case is Double:
print("some other double value that I don't want to print")
case let someString as String:
print("a string value of \"\(someString)\"")
case let (x, y) as (Double, Double):
print("an (x, y) point at \(x), \(y)")
case let movie as Movie:
print("a movie called \(movie.name), dir. \(movie.director)")
case let stringConverter as (String) -> String:
print(stringConverter("Michael"))
default:
print("something else")
}
}
// zero as an Int
// zero as a Double
// an integer value of 42
// a positive double value of 3.14159
// a string value of "hello"
// an (x, y) point at 3.0, 5.0
// a movie called Ghostbusters, dir. Ivan Reitman
// Hello, Michael
注意
Any类型表示了任意类型的值,包括可选类型。如果你给显式声明的Any类型使用可选项,Swift 就会发出警告。如果你真心需要在Any值中使用可选项,如下所示,你可以使用as运算符来显式地转换可选项为Any。
let optionalNumber: Int? = 3
things.append(optionalNumber) // Warning
things.append(optionalNumber as Any) // No warning