在swift 3中新增加了兩種訪問控制權限fileprivate
和open
镀裤。
private
在原有的swift中的private其實并不是真正的私有销部,如果一個變量定義為private,在同一個文件中的其他類依然是可以訪問到的奥帘。這個場景在使用extension的時候很明顯铜邮。
class Person {
private var name = "jack"
}
extension User{
var accessPrivate: String {
return name
}
}
這樣帶來了兩個問題:
當我們標記為private
時,意為真的私有還是文件內可共享呢?
當我們如果意圖為真正的私有時松蒜,必須保證這個類或者結構體在一個單獨的文件里扔茅。否則可能同文件里其他的代碼訪問到。
由此秸苗,在swift 3中召娜,新增加了一個fileprivate
來顯式的表明,這個元素的訪問權限為文件內私有惊楼。
過去的private對應現(xiàn)在的fileprivate【寥常現(xiàn)在的private則是真正的私有,離開了這個類或者結構體的作用域外面就無法訪問檀咙。
所以fileprivate > private .
open
open則是彌補public語義上的不足雅倒。
現(xiàn)在的pubic有兩層含義:
這個元素可以在其他作用域被訪問
這個元素可以在其他作用域被繼承或者override
繼承是一件危險的事情。尤其對于一個framework或者module的設計者而言弧可。在自身的module內蔑匣,類或者屬性對于作者而言是清晰的,能否被繼承或者override都是可控的棕诵。但是對于使用它的人裁良,作者有時會希望傳達出這個類或者屬性不應該被繼承或者修改。這個對應的就是final校套。
final的問題在于在標記之后价脾,在任何地方都不能override。而對于lib的設計者而言搔确,希望得到的是在module內可以被override彼棍,在被import到其他地方后其他用戶使用的時候不能被override。
這就是open產生的初衷膳算。通過open和public標記區(qū)別一個元素在其他module中是只能被訪問還是可以被override。
通俗的理解public和open就是:
public:可以被任何人訪問弛作,但其他module中不可以被override和繼承涕蜂,而在本module內可以被override和繼承。
open:可以被任何人使用映琳,包括override和繼承机隙。
For example
/// ModuleA:
// 這個類在ModuleA的范圍外是不能被繼承的,只能被訪問
public class NonSubclassableParentClass {
public func foo() {}
// 這是錯誤的寫法萨西,因為class已經不能被繼承有鹿,
// 所以他的方法的訪問權限不能大于類的訪問權限
open func bar() {}
// final的含義保持不變
public final func baz() {}
}
// 在ModuleA的范圍外可以被繼承
open class SubclassableParentClass {
// 這個屬性在ModuleA的范圍外不能被override
public var size : Int
// 這個方法在ModuleA的范圍外不能被override
public func foo() {}
// 這個方法在任何地方都可以被override
open func bar() {}
///final的含義保持不變
public final func baz() {}
}
/// final的含義保持不變
public final class FinalClass { }
/// ModuleB:
import ModuleA
// 這個寫法是錯誤的,編譯會失敗
// 因為NonSubclassableParentClass類訪問權限標記的是public谎脯,只能被訪問不能被繼承
class SubclassA : NonSubclassableParentClass { }
// 這樣寫法可以通過葱跋,因為SubclassableParentClass訪問權限為 `open`.
class SubclassB : SubclassableParentClass {
// 這樣寫也會編譯失敗
// 因為這個方法在SubclassableParentClass 中的權限為public,不是`open'.
override func foo() { }
// 這個方法因為在SubclassableParentClass中標記為open,所以可以這樣寫
// 這里不需要再聲明為open娱俺,因為這個類是internal的
override func bar() { }
}
open class SubclassC : SubclassableParentClass {
// 這種寫法會編譯失敗稍味,因為這個類已經標記為open
// 這個方法override是一個open的方法,則也需要表明訪問權限
override func bar() { }
}
open class SubclassD : SubclassableParentClass {
// 正確的寫法荠卷,方法也需要標記為open
open override func bar() { }
}
open class SubclassE : SubclassableParentClass {
// 也可以顯式的指出這個方法不能在被override
public final override func bar() { }
}
For example 2
//FirstDog.swift
class FirstDog: NSObject {
//name屬性
//不加任何修飾 本類和其他類都可以訪問dogName
var dogName : String = "fitstDog"
//private絕對私有模庐,其他類中不能訪問,本類的extension中也不能訪問油宜,子類繼承父類也不能訪問
// error:'dogName' is inaccessible due to 'private' protection level
private var dogName : String = "fitstDog"
//fileprivate是文件私有掂碱,一個文件是可以有多個類的,標記為fileprivate的常理在多個類里面都可以用慎冤。
//換句話說疼燥,在當前FirstDog類 文件中 都可以使用,換成其他類不能使用粪薛,繼承也不能使用
//'dogName' is inaccessible due to 'fileprivate' protection level
fileprivate var dogName : String = "fitstDog"
//這個寫法就是set方法私有
//extension悴了、繼承、其他類违寿,賦值dogName的操作都會報錯
//error:Cannot assign to property: 'dogName' setter is inaccessible
private(set) var dogName : String = "fitstDog"
func temp1() {
//不加任何修飾 本類湃交、其他類、extension中 都可以訪問dogName
dogName = "fitstCat"
}
}
//本類的 extension
extension FirstDog {
func temp2() {
//修改
dogName = "fitstCat"
print(dogName)
}
}
//FirstDog.swift
// 本類FirstDog文件中的 另一個類ThirdDog
// fileprivate是文件私有藤巢,一個文件是可以有多個類的搞莺,標記為fileprivate的常理在多個類里面都可以用,但如果標記為private掂咒,在不同類是不能訪問的才沧。
class ThirdDog: NSObject {
let fitstDog = FirstDog()
func temp3() {
//修改
fitstDog.dogName = "thirdDog"
print(fitstDog.dogName)
}
}
//FirstDog.swift
// 本類FirstDog文件中的 另一個類FourthDog 繼承FirstDog
//繼承父類FirstDog 如果屬性dogName設置為private 子類FourthDog是不能訪問的
class FourthDog: FirstDog {
let fitstDog = FirstDog()
func temp4() {
//修改
fitstDog.dogName = "thirdDog"
print(fitstDog.dogName)
}
}
SecondDog.swift
import UIKit
class SecondDog: NSObject {
let firstDog = FirstDog()
func temp() {
print(firstDog.dogName)
firstDog.dogName = "secondDog"
}
}
總結:
private私有,本類的extension绍刮、其他類温圆、繼承都不能使用
fileprivate文件私有,本類文件內可以使用孩革,文件外其他類岁歉、繼承都不能使用
private(set) set方法私有,不能修改