附属脚本
可以定义在类(Class)、结构体(structure)和枚举(enumeration)这些目标中,可以认为是访问对象、集合或序列的快捷方式,不需要再调用实例的特定的赋值和访问方法
附属脚本语法
附属脚本允许你通过在实例后面的方括号中传入一个或者多个的索引值来对实例进行访问和赋值。语法类似于实例方法和计算型属性的混合。与定义实例方法类似,定义附属脚本使用subscript关键字,显式声明入参(一个或多个)和返回类型。与实例方法不同的是附属脚本可以设定为读写或只读。这种方式又有点像计算型属性的getter和setter
subscript(index: Int) -> Int { get { // 返回与入参匹配的Int类型的值 } set(newValue) { // 执行赋值操作 } }
newValue的类型必须和附属脚本定义的返回类型相同,与计算型属性相同的是set的入参声明newValue就算不写,在set代码块中依然可以使用默认的newValue这个变量来访问新赋的值。
下面代码演示了一个在TimesTable结构体中使用只读附属脚本的用法,该结构体用来展示传入整数的n倍。
struct TimesTable { let multiplier: Int subscript(index: Int) -> Int { return multiplier * index } } let threeTimesTable = TimesTable(multiplier: 3) //threeTimesTable[6],这句话访问了threeTimesTable的第六个元素,返回18或者6的3倍。 println("3的6倍是\(threeTimesTable[6])") // 输出 "3的6倍是18"
附属脚本用法
据使用场景不同附属脚本也具有不同的含义。通常附属脚本是用来访问集合(collection),列表(list)或序列(sequence)中元素的快捷方式。
Swift 的字典(Dictionary)实现了通过附属脚本来对其实例中存放的值进行存取操作。在附属脚本中使用和字典索引相同类型的值,并且把一个字典值类型的值赋值给这个附属脚本来为字典设值
var numberOfLegs = ["spider": 8,"ant": 6,"cat": 4] numberOfLegs["bird"] = 2上例定义一个名为 numberOfLegs 的变量并用一个字典字面量初始化出了包含三对键值的字典实例。 numberOfLegs 的字典存放值类型推断为 Dictionary<String,Int> 。字典实例创建完成之后通过附属脚本的方式将整型值 2 赋值到字典实例的索引为 bird 的位置中。
附属脚本选项
附属脚本允许任意数量的入参索引,并且每个入参类型也没有限制。附属脚本的返回值也可以是任何类型。附属脚本可以使用变量参数和可变参数,但使用写入读出(in-out)参数或给参数设置默认值都是不允许的。
一个类或结构体可以根据自身需要提供多个附属脚本实现,在定义附属脚本时通过入参个类型进行区分,使用附属脚本时会自动匹配合适的附属脚本实现运行,这就是附属脚本的重载。
一个附属脚本入参是最常见的情况,但只要有合适的场景也可以定义多个附属脚本入参。如下例定义了一个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,"Index out of range") grid[(row * columns) + columns] = newValue } } }Matrix 提供了一个两个入参的构造方法,入参分别是 rows 和 columns ,创建了一个足够容纳 rows * columns 个数的 Double 类型数组。为了存储,将数组的大小和数组每个元素初始值 0.0 ,都传入数组的构造方法中来创建一个正确大小的新数组。
var matrix = Matrix(rows: 2,columns: 2)
为了方便进行断言,Matrix包含了一个名为indexIsValid的成员方法,用来确认入参的row或column值是否会造成数组越界:
断言在附属脚本越界时触发:
let someValue = matrix[2,2]
断言将会触发,因为 [2,2] 已经超过了matrix的最大长度