Swift 之訪問控制

imwu7164 7年前發布 | 31K 次閱讀 Swift Apple Swift開發

訪問控制對訪問你的其他代碼源文件和模塊部分進行了約束。這個特性允許你隱藏你的代碼實現,并且指定通過其可以訪問和使用該代碼的優選接口。

class,structure 和 enumeration 都可以指定訪問級別,當然,property,method,initializer 和 屬于這里類型的 subscript。protocol 可以限制到某個上下文,全局變量,變量和函數也可以。

另外,Swift 也提供默認的使用級別給典型的使用場景。確實,如果你編寫一款單一目標的 app,你可能根本不需要明確地指定訪問控制級別。

模塊和源文件

Swift的訪問控制模型是基于模塊和源文件的概念。

一個模塊是單個的代碼分布單元————一個 framework 或者應用程序是作為單個單元編譯和傳遞的,他們能夠通過 Swift 的 import 關鍵字被其他模塊導入。

在Swift 中, Xcode的每一個 build target(如 一個 app bundle 或者 framework)被當成一個單獨的模塊。

雖然通常做法是在不同的源文件定義不同的類型,然而一個源文件事實上可以包含不同的類型,函數等的定義。

訪問級別

Swift 為你的代碼實體提供5個不同的訪問級別:

  • Open 訪問和 public 訪問允許實體能夠被使用在任何來自起決定作用的模塊的源文件,或者來自于其他被導入的模塊的源文件。通常使用 open 或者 public 來指定framework 的公開接口。兩者的不同點將在下面進行描述。

  • Internal 訪問允許實體被使用在他們定義模型的任何源文件里面,但是不能在模塊外部的任何源文件使用。通常在定義一個 app 或者一個 framework 的內部結構的時候使用 internal 訪問。

  • File-private 訪問限制了在定義源文件中實體的使用。使用 file-private 訪問來隱藏特定功能的實現細節,當這些細節在整個文件中使用的時候。

  • Private 訪問將實體的使用限制在封閉聲明中。使用 private 訪問來隱藏特定功能的實現細節,當這些細節在單個聲明使用時。

Open 訪問是最高訪問級別,private 是最低訪問級別(最大限制性)。

Open 訪問只用在類和類成員,他和 publick 訪問的區別如下:

  • 使用 public 訪問的類, 或者其他更多限制性的訪問級別,只能在定義的模塊內創建子類。

  • 使用 public 訪問的類成員,或者其他更多限制性的訪問級別,只能在定義的模塊內被其子類重寫。

  • Open 類可以被定義的模塊或者其他 import 該模塊的地方創建子類。

  • Open 類成員可以被定義的模塊或者其他 import 該模塊的地方創建的子類重寫。

簡單來說就是 public 和 open 的區別就是public 比 open 少了模塊外的類繼承和類成員重寫的權限。

訪問級別的指導原則

在 Swift 中,訪問級別遵從總的指導原則是:沒有實體可以被定義在另外一個擁有較低訪問級別(更多限制)的實體之內。

例如:

  • public變量不能被定義為具有internal, file-private或者 private 類型,因為這種類型可能不能用在使用公共變量的任何地方。

  • 函數不能具有比其他參數類型和返回類型更高的訪問級別,因為該函數可以在其組成類型不可被周圍代碼使用的情況下使用。

下面會有更詳細的介紹。

默認的訪問級別

如果你自己沒有指定一個明確的訪問級別,所有代碼中的實體都有一個默認的internal訪問級別。結果,在很多情況下你不需要對你的代碼指定明確的訪問級別。

單目標應用程序的訪問級別

如果你寫的是一個 i 簡單的單目標應該程序,那么你的程序代碼就是典型的自包含程序,并不需要在程序模塊的外部進行使用。默認的訪問級別 internal 已經滿足這個需求。因此,你不需要去指定一個訪問級別。然而,你可能需要把你部分的代碼標記為文件私有或者私有,從而使得在程序模塊中的其他代碼隱藏他們的實現細節。

Frameworks 的訪問級別

當你開發一個 framework,標記 open 或者 public 以便它能夠被其他模塊訪問到,例如某個程序引入這個 framework 的時候。這個面向公眾的接口是framework 的程序編程接口(或者 API)。

值得注意的是:任何 framework 的內部實現細節都還可以使用默認的內部訪問級別,或者可以標識為私有或者文件私有級別,如果你想對framework 的其他部分內部代碼隱藏他們的話。只有當你想讓一個實體成為你的 framework 的 API 的一部分的話,那么你就需要把這個實體標識為 open 或者 public。

單元測試目標的訪問級別

當你寫的是一個包含單元測試目標的程序時,那么你需要讓你程序中的代碼可以被測試模塊使用到以便于測試。一般情況下,只有被標識為 open 或者 public 的實體才可以被其他模塊訪問到。然而,如果你把產品的模塊 import 聲明前加入 @testable 屬性并且在打開測試選項下編譯產品模塊的話,那么單元測試目標就能夠訪問任何的 internal 實體。

訪問級別語法

為實體定義訪問級別:

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() {}

除非有其他的指定,否則的話默認的訪問基本是 internal,這也就意味著 SomeInternalClass 和 someInternalConstant 能夠在不明確訪問級別修飾符的情況下也還擁有 internal 的訪問級別:

class SomeInternalClass{}              // 隱式 internal
let someInternalConstant = 0            // 隱式 internal

翻譯自: https://developer.apple.com/library/content/documentation/Swift/Conceptual/Swift_Programming_Language/AccessControl.html#//apple_ref/doc/uid/TP40014097-CH41-ID3

The End

 

來自:https://devlong.com/2017/08/12/swift-access-control.md/

 

 本文由用戶 imwu7164 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!