Swift3.0中文教程:12.下标
2016-12-08 18:15
435 查看
下标 (subscripts)可以定义在类(class)、结构体(structure)和枚举(enumeration)中,是访问集合(collection),列表(list)或序列(sequence)中元素的快捷方式。可以使用下标的索引,设置和获取值,而不需要再调用对应的存取方法。举例来说,用下标访问一个
一个类型可以定义多个下标,通过不同索引类型进行重载。下标不限于一维,你可以定义具有多个入参的下标满足自定义类型的需求。
如同只读计算型属性,可以省略只读下标的
下面代码演示了只读下标的实现,这里定义了一个
在上例中,创建了一个
你可以通过下标访问
注意
例如,Swift 的
上例定义一个名为
更多关于
注意
Swift 的
一个类或结构体可以根据自身需要提供多个下标实现,使用下标时将通过入参的数量和类型进行区分,自动匹配合适的下标,这就是下标的重载。
虽然接受单一入参的下标是最常见的,但也可以根据情况定义接受多个入参的下标。例如下例定义了一个
你可以通过传入合适的
上例中创建了一个
![](https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/Swift_Programming_Language/Art/subscriptMatrix01_2x.png)
将
上面两条语句分别调用下标的 setter 将矩阵右上角位置(即
![](https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/Swift_Programming_Language/Art/subscriptMatrix02_2x.png)
断言在下标越界时触发:
Array实例中的元素可以写作
someArray[index],访问
Dictionary实例中的元素可以写作
someDictionary[key]。
一个类型可以定义多个下标,通过不同索引类型进行重载。下标不限于一维,你可以定义具有多个入参的下标满足自定义类型的需求。
下标语法
下标允许你通过在实例名称后面的方括号中传入一个或者多个索引值来对实例进行存取。语法类似于实例方法语法和计算型属性语法的混合。与定义实例方法类似,定义下标使用subscript关键字,指定一个或多个输入参数和返回类型;与实例方法不同的是,下标可以设定为读写或只读。这种行为由 getter 和 setter 实现,有点类似计算型属性:
subscript(index: Int) -> Int { get { // 返回一个适当的 Int 类型的值 } set(newValue) { // 执行适当的赋值操作 } }
newValue的类型和下标的返回类型相同。如同计算型属性,可以不指定 setter 的参数(
newValue)。如果不指定参数,setter 会提供一个名为
newValue的默认参数。
如同只读计算型属性,可以省略只读下标的
get关键字:
subscript(index: Int) -> Int { // 返回一个适当的 Int 类型的值 }
下面代码演示了只读下标的实现,这里定义了一个
TimesTable结构体,用来表示传入整数的乘法表:
struct TimesTable { let multiplier: Int subscript(index: Int) -> Int { return multiplier * index } } let threeTimesTable = TimesTable(multiplier: 3) print("six times three is \(threeTimesTable[6])") // 输出 "six times three is 18"
在上例中,创建了一个
TimesTable实例,用来表示整数
3的乘法表。数值
3被传递给结构体的构造函数,作为实例成员
multiplier的值。
你可以通过下标访问
threeTimesTable实例,例如上面演示的
threeTimesTable[6]。这条语句查询了
3的乘法表中的第六个元素,返回
3的
6倍即
18。
注意
TimesTable例子基于一个固定的数学公式,对
threeTimesTable[someIndex]进行赋值操作并不合适,因此下标定义为只读的。
下标用法
下标的确切含义取决于使用场景。下标通常作为访问集合(collection),列表(list)或序列(sequence)中元素的快捷方式。你可以针对自己特定的类或结构体的功能来自由地以最恰当的方式实现下标。例如,Swift 的
Dictionary类型实现下标用于对其实例中储存的值进行存取操作。为字典设值时,在下标中使用和字典的键类型相同的键,并把一个和字典的值类型相同的值赋给这个下标:
var numberOfLegs = ["spider": 8, "ant": 6, "cat": 4] numberOfLegs["bird"] = 2
上例定义一个名为
numberOfLegs的变量,并用一个包含三对键值的字典字面量初始化它。
numberOfLegs字典的类型被推断为
[String: Int]。字典创建完成后,该例子通过下标将
String类型的键
bird和
Int类型的值
2添加到字典中。
更多关于
Dictionary下标的信息请参考读取和修改字典
注意
Swift 的
Dictionary类型的下标接受并返回可选类型的值。上例中的
numberOfLegs字典通过下标返回的是一个
Int?或者说“可选的int”。
Dictionary类型之所以如此实现下标,是因为不是每个键都有个对应的值,同时这也提供了一种通过键删除对应值的方式,只需将键对应的值赋值为
nil即可。
下标选项
下标可以接受任意数量的入参,并且这些入参可以是任意类型。下标的返回值也可以是任意类型。下标可以使用变量参数和可变参数,但不能使用输入输出参数,也不能给参数设置默认值。一个类或结构体可以根据自身需要提供多个下标实现,使用下标时将通过入参的数量和类型进行区分,自动匹配合适的下标,这就是下标的重载。
虽然接受单一入参的下标是最常见的,但也可以根据情况定义接受多个入参的下标。例如下例定义了一个
Matrix结构体,用于表示一个
Double类型的二维矩阵。
Matrix结构体的下标接受两个整型参数:
struct Matrix { let rows: Int, columns: Int var grid: [Double] init(rows: Int, columns: Int) { self.rows = rows self.columns = columns grid = Array(count: rows * columns, repeatedValue: 0.0) } func indexIsValidForRow(row: Int, column: Int) -> Bool { return row >= 0 && row < rows && column >= 0 && column < columns } subscript(row: Int, column: Int) -> Double { get { assert(indexIsValidForRow(row, column: column), "Index out of range") return grid[(row * columns) + column] } set { assert(indexIsValidForRow(row, column: column), "Index out of range") grid[(row * columns) + column] = newValue } } }
Matrix提供了一个接受两个入参的构造方法,入参分别是
rows和
columns,创建了一个足够容纳
rows * columns个
Double类型的值的数组。通过传入数组长度和初始值
0.0到数组的构造器,将矩阵中每个位置的值初始化为
0.0。关于数组的这种构造方法请参考创建一个空数组。
你可以通过传入合适的
row和
column的数量来构造一个新的
Matrix实例:
var matrix = Matrix(rows: 2, columns: 2)
上例中创建了一个
Matrix实例来表示两行两列的矩阵。该
Matrix实例的
grid数组按照从左上到右下的阅读顺序将矩阵扁平化存储:
![](https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/Swift_Programming_Language/Art/subscriptMatrix01_2x.png)
将
row和
column的值传入下标来为矩阵设值,下标的入参使用逗号分隔:
matrix[0, 1] = 1.5 matrix[1, 0] = 3.2
上面两条语句分别调用下标的 setter 将矩阵右上角位置(即
row为
0、
column为
1的位置)的值设置为
1.5,将矩阵左下角位置(即
row为
1、
column为
0的位置)的值设置为
3.2:
![](https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/Swift_Programming_Language/Art/subscriptMatrix02_2x.png)
Matrix下标的 getter 和 setter 中都含有断言,用来检查下标入参
row和
column的值是否有效。为了方便进行断言,
Matrix包含了一个名为
indexIsValidForRow(_:column:)的便利方法,用来检查入参
row和
column的值是否在矩阵范围内:
func indexIsValidForRow(row: Int, column: Int) -> Bool { return row >= 0 && row < rows && column >= 0 && column < columns }
断言在下标越界时触发:
let someValue = matrix[2, 2] // 断言将会触发,因为 [2, 2] 已经超过了 matrix 的范围
相关文章推荐
- Swift3.0中文教程:5.控制流
- Swift3.0中文教程:15.析构过程
- Swift3.0中文教程: 8.枚举
- Swift3.0中文教程:11.方法
- Swift3.0中文教程:21.拓展
- Swift中文教程(十二) 下标
- Swift3.0中文教程:13.继承
- Swift3.0中文教程:23.泛型
- Swift3.0中文教程:22.协议
- Swift3.0中文教程: 9.类和结构体
- Swift3.0中文教程:4.集合类型
- Swift中文基础教程----下标
- Swift3.0中文教程:19.类型转换
- Silverlight 3.0中文教程
- Expression Blend实例中文教程(12) - 样式和模板快速入门Style,Template
- Installshield 12 中文系列教程之自定义安装行
- Varnish3.0中文入门教程
- Installshield 12 中文系列教程之 配置服务 (三)配置IIS
- Expression Blend实例中文教程(12) - 样式和模板快速入门Style,Template
- Expression Blend实例中文教程(12) - 样式和模板快速入门Style,Template