存取控制

Swift 提供存取控制(access control)的特性,讓你可以為程式碼或模組設置存取權限,決定哪些部分可以開放給外部程式碼使用。

Swift 中,可以為列舉、結構或類別設置存取權限,這些型別內部的屬性、方法、下標或建構器也可以設置存取權限。而協定中的全域變數、常數或函式也可以設置存取權限。

如果你只是單純的在開發一個獨立的應用程式,而不是開發一個模組,其實可以不用顯式地設置存取權限,Swift 已為大部分的情況提供預設存取權限。

Hint
  • Swift 的存取控制主要是基於模組(module)和源文件(source file)這兩個概念。
  • 模組指的是一個獨立的程式碼組件,在 Swift 中,可以在一個模組中使用import關鍵字來引用另一個模組。以 Xcode 來說,每個 target (也就是框架framework或應用程式application)都是一個獨立的模組。另外如果將一些通用功能或常用程式打包成獨立的框架,這也是一種模組。
  • 源文件表示一個 Swift 的源程式文件( Swift source code file ),通常會屬於在一個模組內。

存取層級

Swift 提供了 5 種不同的存取層級,分別如下:

  • Open:公開存取的層級,同模組中的任何程式及外部引用這個模組的程式都可以存取繼承類別或是覆寫類別成員(即屬性或方法)。
  • Public:公開存取的層級,以存取來說,規則與 Open 層級一樣。不同的是,只有在其定義的同模組內的程式可以繼承類別或是覆寫類別成員,外部引用這個模組的程式則無法。
  • Internal:內部存取的層級,同模組的源文件中的程式可以使用,但外部引用的程式不能使用。當定義為只供應用程式或模組內部使用時,可以將其設為internal
  • File-private:私有存取的層級,只能在原始定義的源文件中使用。使用fileprivate可以用來隱藏特定功能的實作。
  • Private:私有存取的層級,只能在原始定義的程式區域中(像是一個類別內)使用。使用private可以用來隱藏特定功能的實作。

使用方式為在變數、函式或類別的前面加上這 5 種不同的關鍵字來宣告它們的存取層級,如下:

public class SomePublicClass {}
internal class SomeInternalClass {}
fileprivate class SomeFilePrivateClass {}
private class SomePrivateClass {}

public var somePublicVariable = 0
internal let someInternalConstant = 0
fileprivate func someFilePrivateFunction() {}
private func somePrivateFunction() {}

存取層級基本原則

Swift 存取層級的基本原則:不可以在一個實體中定義存取層級限制更嚴格的實體。如以下的例子:

  • 一個public的變數,不能將它的型別定義為internalfileprivateprivate。因為當變數可以被公開存取,但定義它的型別不行,這樣會出現錯誤。
  • 一個函式的參數、返回型別存取層級限制不能比函式本身的更嚴格。因為當函式存取層級設為public,可以被公開存取時,但參數或返回型別存取層級設為internalfileprivateprivate,無法被公開存取,這樣會出現錯誤。

預設存取層級

如果沒有顯性宣告存取層級時,internal就是預設的存取層級。

自定義型別

你也可以為自定義的型別定義存取層級,當然你必須確認這個型別的作用範圍與存取層級的限制相符。

類別的存取層級會影響其內部成員的存取層級,像是定義一個private的類別,則其內部成員的存取層級都會是private。而定義一個publicinternal的類別,其內部成員都會是internal

Hint
  • 一個public類別,其內成員預設存取層級為internal而不是public,如果要為某個成員設為public,必須顯式定義。這樣在你定義一個模組的公開介面時,可以明確的選擇哪些介面是公開的,避免不小心將內部使用的介面公開。

以下是幾個類別與其內部成員存取層級的例子:

// 顯式指定為 public 類別
public class SomePublicClass {
    // 顯式指定為 public 成員
    public var somePublicProperty = 0

    // 隱式推斷為 internal 成員
    var someInternalProperty = 0

    // 顯式指定為 fileprivate 成員
    fileprivate func someFilePrivateMethod() {}

    // 顯式指定為 private 成員
    private func somePrivateMethod() {}
}

// 隱式推斷為 internal 類別
class SomeInternalClass {
    // 隱式推斷為 internal 成員
    var someInternalProperty = 0

    // 顯式指定為 fileprivate 成員
    fileprivate func someFilePrivateMethod() {}

    // 顯式指定為 private 成員
    private func somePrivateMethod() {}
}

// 顯式指定為 private 類別
private class SomePrivateClass {
    // 隱式推斷為 private 成員
    var somePrivateProperty = 0

    // 隱式推斷為 private 成員
    func somePrivateMethod() {}
}

元組型別

元組不能明確指定存取層級,而是在使用時被自動推斷的。元組的存取層級會根據其內成員存取層級最嚴格的一個為準,例子如下:

private var description = "Sunny day !"
internal var number = 300
public var name = "Joe Black"

// 這時這個元組的存取層級會根據最嚴格的 private 為準
let someTuple = (description, number, name)

函式型別

函式的存取層級是根據存取層級最嚴格的參數或返回值型別來決定。經由此規則得到的存取層級如果與預設存取層級不一樣,則必須明確的指定函式的存取層級,如下:

// 定義一個用來當做 函式返回值型別 的類別
private class SomeClass {}

// 定義一個函式
// 返回值為 SomeClass 存取層級為 private
// 則這個函式的存取層級也為 private
// 這時與預設的 internal 不一樣 所以必須明確指定函式為 private
// 如果將函式前面的 private 拿掉 會報錯誤
private func someFunction() -> SomeClass {
    return SomeClass()
}

列舉型別

列舉成員的存取層級與列舉相同,無法為列舉成員單獨指定不同的存取層級,如下:

// 定義列舉的存取層級為 public
public enum CompassPoint {
    // 則列舉成員都為 public
    case north
    case south
    case east
    case west
}

而列舉的原始值(raw value)與相關值(associated value)的存取層級限制不能比列舉的存取層級嚴格。例如,你不能在一個internal的列舉中定義private的原始值。

巢狀型別

以巢狀型別來說,定義在private型別中的巢狀型別,會自動指定為private。而在publicinternal型別中,巢狀型別則自動指定為internal,這時如果要指定巢狀型別為public,則必須明確指定為public

子類別

子類別的存取層級限制不能比父類別更為寬鬆,例如父類別為internal時,子類別就不能是public

在符合當前存取層級限制的條件下:

  • 子類別可以覆寫父類別任意的成員(方法、屬性、建構器或下標等),用來提供限制較寬鬆的存取層級。
  • 子類別成員可以存取限制更嚴格的父類別成員(因為子類別與父類別定義在同一個源文件中)。

以下是一個例子:

// 定義一個 public 的類別 A
public class A {
    // 定義一個 private 的方法
    private func someMethod() {}
}

// 繼承自 A 的類別 B 其存取層級為 internal
// 符合 子類別的存取層級限制不能比父類別更為寬鬆
internal class B: A {
    // 可以覆寫父類別的方法 更新為較寬鬆的存取層級
    // (當然必須符合自身的存取層級)
    override internal func someMethod() {
        // 可以呼叫 存取層級限制更嚴格的父類別成員
        super.someMethod()
    }
}

常數, 變數, 屬性及下標

常數、變數及屬性不能擁有比它們的型別限制更為寬鬆的存取層級。例如,不能定義一個public的屬性,但它的型別卻是private。而下標也不能擁有比它們的索引值或返回值型別限制更為寬鬆的存取層級。以下是一個例子:

// 定義一個 private 的類別
private class SomeClass {}

// 這時變數的型別為 private 則變數必須顯式的定義為 private
// 將 private 拿掉會報錯誤 因為會變成預設的 internal 則與規則不符
private var someInstance = SomeClass()

Getter 與 Setter

常數、變數、下標與屬性的GetterSetter的存取層級與他們所屬型別的存取層級相同。

Setter的存取層級可以比對應的Getter存取層級限制更為嚴格,可以用來控制其讀寫權限。使用方式為在varsubscript關鍵字前,加上private(set)internal(set)來指定限制更為嚴格的存取層級。以下是一個例子:

// 定義一個結構 預設存取層級為 internal
struct TrackedString {
    // 將變數的 Setter 存取層級指定為 private
    private(set) var numberOfEdits = 0
    var value: String = "" {
        didSet {
            // 所以在結構內部 是可以讀寫的
            numberOfEdits += 1
        }
    }
}

// 宣告一個結構的變數
var stringToEdit = TrackedString()

// 每修改一次 會經由屬性觀察器來將內部變數屬性加一
stringToEdit.value = "字串修改次數會被記錄"
stringToEdit.value += "每修改一次, numberOfEdits 數字會加一"
stringToEdit.value += "這行修改也會加一"

// 印出:已修改了 3 次
print("已修改了 \(stringToEdit.numberOfEdits) 次")

你也可以在必要時為GetterSetter顯式指定存取層級,例子如下:

// 顯式指定這個結構的存取層級為 public
public struct TrackedString {
    // 結合 public 與 private(set)
    // 所以這時這個變數屬性的 Setter 為 private
    // Getter 為 public
    public private(set) var numberOfEdits = 0
    public var value: String = "" {
        didSet {
            numberOfEdits += 1
        }
    }
    public init() {}
}

建構器

自定義建構器的存取層級限制不能比其所屬型別的存取層級寬鬆,像是一個internal的類別,不能設置一個public的建構器。而唯一例外是,當建構器為必要建構器(required initializer)時,其存取層級必須與所屬型別相同。

與函式或方法一樣,建構器參數的存取層級限制也不能比建構器本身嚴格。像是一個internal的建構器,不能設置一個private的參數。

預設建構器

預設建構器的存取層級與所屬型別的存取層級相同。除非當型別的存取層級為public,則預設建構器會被設置為internal,如果需要一個public的建構器,必須自己定義一個,如下:

public class SomeClass {
    public init() {}
}

結構的成員逐一建構器

如果結構中任意儲存屬性的存取層級為private,那麼這個結構預設的成員逐一建構器的存取層級就是private,否則就為internal

如果需要在其他模組也可以使用這個結構的成員逐一建構器,則必須自行定義一個public的成員逐一建構器。

協定

如果想為一個協定明確的指定存取層級,必須在定義此協定時指定。這樣可以確保這個協定只能在適當的存取層級範圍內被遵循。

協定中的每個功能都與該協定的存取層級相同,這樣才能確保協定所有功能都可以被遵循此協定的型別存取。

協定繼承

從已存在的協定繼承了一個新的協定時,這個新協定的存取層級不能比已存在協定的寬鬆。例如,定義一個public的協定時,不能繼承自一個internal的協定。

協定一致性

一個型別可以遵循一個存取層級限制更為嚴格的協定,例如,你可以定義一個public的型別,並遵循一個internal的協定。

遵循了協定的型別的存取層級,會以型別本身遵循的協定限制較嚴格的存取層級為準,如果一個public的型別,遵循了一個internal的協定,則在此型別作為符合協定的型別時,其存取層級也是internal

當你讓一個型別遵循某個協定並滿足其所有要求後,你必須確保所有這些實作協定的部分,其存取層級不能比協定更為嚴格。例如一個public的型別,遵循了internal的協定,則實作協定的部份最嚴格只能到internal存取層級。

擴展

你可以在符合存取層級的情況下,擴展一個列舉、結構或類別,這個擴展會與擴展的對象擁有一樣的存取層級。例如,你擴展了一個publicinternal型別,擴展中的成員則預設為internal,與原始型別中的成員一樣。而當擴展了一個private型別時,擴展成員則預設為private

或者,你也可以明確的指定擴展的存取層級,來讓其內成員都預設成一樣的存取層級。這個預設的存取層級仍可被個別成員所指定的存取層級覆蓋。

經由擴展來遵循協定

如果你經由擴展來遵循協定,那你就不能顯式的指定這個擴展的存取層級了。而這個協定本身的存取層級會變成預設的存取層級,且擴展中每個協定功能的實作也是一樣的預設的存取層級。

泛型

泛型型別或泛型函式的存取層級由泛型型別或泛型函式本身泛型的型別約束參數中限制最嚴格的來確定。

型別別名

自定義的任何型別別名都會被當做不同的型別來做存取控制。型別別名不能擁有比原始型別限制更為寬鬆的存取層級。

  • 一個public的型別,可以宣告為privatefileprivateinternalpublic的型別別名。
  • 一個private的型別,僅能宣告為private的型別別名,不能宣告為fileprivateinternalpublicopen的型別別名。

results matching ""

    No results matching ""