Swift 下標(biāo)腳本

下標(biāo)腳本 可以定義在類(Class)、結(jié)構(gòu)體(structure)和枚舉(enumeration)這些目標(biāo)中,可以認(rèn)為是訪問對象、集合或序列的快捷方式,不需要再調(diào)用實(shí)例的特定的賦值和訪問方法。

舉例來說,用下標(biāo)腳本訪問一個(gè)數(shù)組(Array)實(shí)例中的元素可以這樣寫 someArray[index] ,訪問字典(Dictionary)實(shí)例中的元素可以這樣寫 someDictionary[key]。

對于同一個(gè)目標(biāo)可以定義多個(gè)下標(biāo)腳本,通過索引值類型的不同來進(jìn)行重載,而且索引值的個(gè)數(shù)可以是多個(gè)。


下標(biāo)腳本語法及應(yīng)用

語法

下標(biāo)腳本允許你通過在實(shí)例后面的方括號(hào)中傳入一個(gè)或者多個(gè)的索引值來對實(shí)例進(jìn)行訪問和賦值。

語法類似于實(shí)例方法和計(jì)算型屬性的混合。

與定義實(shí)例方法類似,定義下標(biāo)腳本使用subscript關(guān)鍵字,顯式聲明入?yún)ⅲㄒ粋€(gè)或多個(gè))和返回類型。

與實(shí)例方法不同的是下標(biāo)腳本可以設(shè)定為讀寫或只讀。這種方式又有點(diǎn)像計(jì)算型屬性的getter和setter:

subscript(index: Int) -> Int {
    get {
        // 用于下標(biāo)腳本值的聲明
    }
    set(newValue) {
        // 執(zhí)行賦值操作
    }
}

實(shí)例 1

import Cocoa

struct subexample {
    let decrementer: Int
    subscript(index: Int) -> Int {
        return decrementer / index
    }
}
let division = subexample(decrementer: 100)

print("100 除以 9 等于 \(division[9])")
print("100 除以 2 等于 \(division[2])")
print("100 除以 3 等于 \(division[3])")
print("100 除以 5 等于 \(division[5])")
print("100 除以 7 等于 \(division[7])")

以上程序執(zhí)行輸出結(jié)果為:

100 除以 9 等于 11
100 除以 2 等于 50
100 除以 3 等于 33
100 除以 5 等于 20
100 除以 7 等于 14

在上例中,通過 subexample 結(jié)構(gòu)體創(chuàng)建了一個(gè)除法運(yùn)算的實(shí)例。數(shù)值 100 作為結(jié)構(gòu)體構(gòu)造函數(shù)傳入?yún)?shù)初始化實(shí)例成員 decrementer。

你可以通過下標(biāo)腳本來得到結(jié)果,比如 division[2] 即為 100 除以 2。

實(shí)例 2

import Cocoa

class daysofaweek {
    private var days = ["Sunday", "Monday", "Tuesday", "Wednesday",
        "Thursday", "Friday", "saturday"]
    subscript(index: Int) -> String {
        get {
            return days[index]   // 聲明下標(biāo)腳本的值
        }
        set(newValue) {
            self.days[index] = newValue   // 執(zhí)行賦值操作
        }
    }
}
var p = daysofaweek()

print(p[0])
print(p[1])
print(p[2])
print(p[3])

以上程序執(zhí)行輸出結(jié)果為:

Sunday
Monday
Tuesday
Wednesday

用法

根據(jù)使用場景不同下標(biāo)腳本也具有不同的含義。

通常下標(biāo)腳本是用來訪問集合(collection),列表(list)或序列(sequence)中元素的快捷方式。

你可以在你自己特定的類或結(jié)構(gòu)體中自由的實(shí)現(xiàn)下標(biāo)腳本來提供合適的功能。

例如,Swift 的字典(Dictionary)實(shí)現(xiàn)了通過下標(biāo)腳本對其實(shí)例中存放的值進(jìn)行存取操作。在下標(biāo)腳本中使用和字典索引相同類型的值,并且把一個(gè)字典值類型的值賦值給這個(gè)下標(biāo)腳來為字典設(shè)值:

import Cocoa

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

print(numberOfLegs)

以上程序執(zhí)行輸出結(jié)果為:

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

上例定義一個(gè)名為numberOfLegs的變量并用一個(gè)字典字面量初始化出了包含三對鍵值的字典實(shí)例。numberOfLegs的字典存放值類型推斷為Dictionary。字典實(shí)例創(chuàng)建完成之后通過下標(biāo)腳本的方式將整型值2賦值到字典實(shí)例的索引為bird的位置中。


下標(biāo)腳本選項(xiàng)

下標(biāo)腳本允許任意數(shù)量的入?yún)⑺饕⑶颐總€(gè)入?yún)㈩愋鸵矝]有限制。

下標(biāo)腳本的返回值也可以是任何類型。

下標(biāo)腳本可以使用變量參數(shù)和可變參數(shù)。

一個(gè)類或結(jié)構(gòu)體可以根據(jù)自身需要提供多個(gè)下標(biāo)腳本實(shí)現(xiàn),在定義下標(biāo)腳本時(shí)通過傳入?yún)?shù)的類型進(jìn)行區(qū)分,使用下標(biāo)腳本時(shí)會(huì)自動(dòng)匹配合適的下標(biāo)腳本實(shí)現(xiàn)運(yùn)行,這就是下標(biāo)腳本的重載。

import Cocoa

struct Matrix {
    let rows: Int, columns: Int
    var print: [Double]
    init(rows: Int, columns: Int) {
        self.rows = rows
        self.columns = columns
        print = Array(count: rows * columns, repeatedValue: 0.0)
    }
    subscript(row: Int, column: Int) -> Double {
        get {
            return print[(row * columns) + column]
        }
        set {
            print[(row * columns) + column] = newValue
        }
    }
}
// 創(chuàng)建了一個(gè)新的 3 行 3 列的Matrix實(shí)例
var mat = Matrix(rows: 3, columns: 3)

// 通過下標(biāo)腳本設(shè)置值
mat[0,0] = 1.0
mat[0,1] = 2.0
mat[1,0] = 3.0
mat[1,1] = 5.0

// 通過下標(biāo)腳本獲取值
print("\(mat[0,0])")
print("\(mat[0,1])")
print("\(mat[1,0])")
print("\(mat[1,1])")

以上程序執(zhí)行輸出結(jié)果為:

1.0
2.0
3.0
5.0

Matrix 結(jié)構(gòu)體提供了一個(gè)兩個(gè)傳入?yún)?shù)的構(gòu)造方法,兩個(gè)參數(shù)分別是rows和columns,創(chuàng)建了一個(gè)足夠容納rows * columns個(gè)數(shù)的Double類型數(shù)組。為了存儲(chǔ),將數(shù)組的大小和數(shù)組每個(gè)元素初始值0.0。

你可以通過傳入合適的row和column的數(shù)量來構(gòu)造一個(gè)新的Matrix實(shí)例。