下标

本页包含内容:

[TOC]

下标可以定义在类、结构体和枚举中,是访问集合、列表和序列中元素的快捷方式,可以使用下标的索引,设置和获取值,而不需要再调用对应的存取方法。举例来说,用下标访问一个Array实例中的元素可以写作someArray[index],访问Dictionary实例中的元素可以写作someDictionary[key]

一个类型可以定义多个下标,通过不同索引类型进行重载。下标不限于一维,你可以定义具有多个入参的下标满足自定义类型的需求。

下标语法

下标允许你通过在实例名称后面的方括号中传入一个或者多个索引值类对实例进行存取。语法类似实例方法和计算型语法的混合。与定义实例方法类似,定义下标使用subscript关键字,指定一个或多个输入参数和返回类型;与实例方法不同的使,下标可以设定为读写或只读。这种行为由getter和setter实现,有点类似计算型属性:

subscript(index: Int) -> Int {
    get {
        // 返回一个适当的Int类型值
    }
    set(newValue) {
        // 执行适当的赋值操作
    }
}

newValue的类型和下标的返回类型相同,如同计算型属性,可以不指定setter的参数,如果不指定参数,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的乘法表中的第六个元素,返回36倍即18

注意:TimesTable例子基于一个固定的数学公式,对threeTimesTable[someIndex]进行赋值操作并不合适,因此下标定义为只读的。

下标用法

下标的确切含义取决于使用场景。下标通常作为访问集合、列表和序列中元素的快捷方式。你可以针对自己特定的类或者结构体的功能来自由地义最恰当的方式实现下标。

例如,Swift的Dictionary类型实现下标用于对其实例中存储的值进行存取操作。为字典设置值时,在下标中使用和字典的键类型相同的键,并把一个和字典的值类型的值赋给这个下标:

var numberOfLegs = ["spider": 8, "ant": 6, "cat": 4]
numberOfLegs["bird"] = 2

上例定义一个名为numberOfLegs的变量,并用一个包含三对键值的字典字面量初始化它。numberOfLegs字典的类型被推断为[String: Int]。字典创建完成后,该例子通过下标将String类型的键birdInt类型的值2添加到字典中。

注意: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(repeating: 0.0, count: rows * columns)
    }
    func indexIsValid(row: Int, column: Int) -> Bool {
        return row >=0 && row < rows && column >=0 && column <  columns
    }
    subscript(row: Int, column: Int) -> Double {
        set {
            assert(indexIsValid(row: row, column: column), "Index out of range")
            grid[(row * columns) + column] = newValue
        }
        get {
            assert(indexIsValid(row: row, column: column), "Index out of range")
            return grid[(row * columns) + column]
 
        }
    }
}

Matrix提供了一个接受两个入参的构造方法,入参分别是rowscolumns,创建了一个足够容纳rows * columnsDouble类型的值的数组。通过传入数组长度和初始值0.0到数组的构造器,将矩阵中每个位置的值初始化为0.0

你可以通过传入合适的rowcolumn的数量来构造一个新的Matrix实例:

var matrix = Matrix(rows: 2, columns: 2)

上例中创建了一个Matrix实例来表示两行两列的矩阵。该Matrix实例的grid数组按照从左上到右下的阅读顺序将矩阵扁平化存储:

[图片上传失败...(image-bed4b2-1520405596285)]

rowcolumn的值传入下标来为矩阵设值,下标的入参使用逗号分隔:

matrix[0, 1] = 1.5
matrix[1, 0] = 3.2

上面两条语句分别调用下标的 setter 将矩阵右上角位置(即row0column1的位置)的值设置为1.5,将矩阵左下角位置(即row1column0的位置)的值设置为3.2

[图片上传失败...(image-c75ac0-1520405596285)]

Matrix下标的 getter 和 setter 中都含有断言,用来检查下标入参rowcolumn的值是否有效。为了方便进行断言,Matrix包含了一个名为indexIsValid(row:column:)的便利方法,用来检查入参rowcolumn的值是否在矩阵范围内:

func indexIsValid(row: Int, column: Int) -> Bool {
    return row >= 0 && row < rows && column >= 0 && column < columns
}

断言在下标越界时触发:

let someValue = matrix[2, 2]
// 断言将会触发,因为 [2, 2] 已经超过了 matrix 的范围
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 本章将会介绍 下标语法下标用法下标选项定义一个基类子类生成重写防止重写 下标 下标可以定义在类、结构体和枚举中,是...
    寒桥阅读 465评论 0 1
  • 官方文档 下标脚本 类,结构体和枚举可以定义下标脚本,下标脚本可以认为是访问集合(collection),列表或序...
    hrscy阅读 258评论 1 1
  • //离上次学习swift过了好久,这才刚开始学习swift呀,坚持。 //下标//“下标可以定义在类、结构体和枚举...
    你weixiao的时候很美阅读 591评论 0 0
  • 我想—— 因为想你了 清风已离别相遇的路口 也带不走枯叶一片 只是在茫茫的世界 将深情揉进了秋意 我想—— 因为想...
    静若繁花L阅读 501评论 17 17
  • 似乎我身边离得最近的男性只有我的爸爸、哥哥、弟弟。跟很多女生有蓝颜知己一样,我会跟老爸说很多事情。 每当我听到别人...
    施米诺阅读 426评论 0 0