-
inout
聲明函數(shù)時缨该,在參數(shù)前面用inout修飾,在函數(shù)內(nèi)部實現(xiàn)改變外部參數(shù)川背,注意贰拿,這里只能傳入變量,不能傳入常量和字面量熄云,因為這些是不能變的一旦定義膨更,當(dāng)我們傳入的時候,在變量名字前面用&符號修飾表示缴允,傳遞給inout參數(shù)询一,表明這個變量在參數(shù)內(nèi)部是可以被改變的
注意:inout修飾的參數(shù)是不能有默認(rèn)值的,有范圍的參數(shù)集合也不能被修飾,另外健蕊,一個參數(shù)一旦被inout修飾菱阵,就不能再被var和let修飾了
override func viewDidLoad() {
super.viewDidLoad()
var someInt = 7
var anotherInt = 107
swapTwoInts(&someInt, b: &anotherInt)
print("someInt is \(someInt),anotherInt is \(anotherInt)")
}
func swapTwoInts(inout a:Int , inout b:Int) {
let temp = a
a = b
b = temp
}
-
現(xiàn)在的訪問權(quán)限則依次為:open ,public,internal,fileprivate,private
fileprivate和 open 是新增的兩個關(guān)鍵字-
fileprivate
在原有的swift中的 private其實并不是真正的私有,如果一個變量定義為private缩功,在同一個文件中的其他類依然是可以訪問到的晴及。這個場景在使用extension的時候很明顯。class User { private var name = "private" } extension User{ var accessPrivate: String { return name } }
- 當(dāng)我們標(biāo)記為private時嫡锌,意為真的私有還是文件內(nèi)可共享呢虑稼?
- 當(dāng)我們?nèi)绻鈭D為真正的私有時,必須保證這個類或者結(jié)構(gòu)體在一個單獨的文件里势木。否則可能同文件里其他的代碼訪問到蛛倦。
由此,在swift 3中啦桌,新增加了一個 fileprivate來顯式的表明溯壶,這個元素的訪問權(quán)限為文件內(nèi)私有。過去的private對應(yīng)現(xiàn)在的fileprivate「δ校現(xiàn)在的private則是真正的私有且改,離開了這個類或者結(jié)構(gòu)體的作用域外面就無法訪問。
-
open
open則是彌補public語義上的不足板驳。
現(xiàn)在的pubic有兩層含義:- 這個元素可以在其他作用域被訪問
- 這個元素可以在其他作用域被繼承或者override
繼承是一件危險的事情又跛。尤其對于一個framework或者module的設(shè)計者而言。在自身的module內(nèi)若治,類或者屬性對于作者而言是清晰的慨蓝,能否被繼承或者override都是可控的。但是對于使用它的人端幼,作者有時會希望傳達(dá)出這個類或者屬性不應(yīng)該被繼承或者修改礼烈。這個對應(yīng)的就是 final。
final的問題在于在標(biāo)記之后静暂,在任何地方都不能override济丘。而對于lib的設(shè)計者而言谱秽,希望得到的是在module內(nèi)可以被override洽蛀,在被import到其他地方后其他用戶使用的時候不能被override。
這就是 open產(chǎn)生的初衷疟赊。通過open和public標(biāo)記區(qū)別一個元素在其他module中是只能被訪問還是可以被override郊供。
下面是例子:
-
/// ModuleA:
// 這個類在ModuleA的范圍外是不能被繼承的,只能被訪問
public class NonSubclassableParentClass {
public func foo() {}
// 這是錯誤的寫法近哟,因為class已經(jīng)不能被繼承驮审,
// 所以他的方法的訪問權(quán)限不能大于類的訪問權(quán)限
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類訪問權(quán)限標(biāo)記的是public,只能被訪問不能被繼承
class SubclassA : NonSubclassableParentClass { }
// 這樣寫法可以通過疯淫,因為SubclassableParentClass訪問權(quán)限為 `open`.
class SubclassB : SubclassableParentClass {
// 這樣寫也會編譯失敗
// 因為這個方法在SubclassableParentClass 中的權(quán)限為public地来,不是`open'.
override func foo() { }
// 這個方法因為在SubclassableParentClass中標(biāo)記為open,所以可以這樣寫
// 這里不需要再聲明為open熙掺,因為這個類是internal的
override func bar() { }
}
open class SubclassC : SubclassableParentClass {
// 這種寫法會編譯失敗未斑,因為這個類已經(jīng)標(biāo)記為open
// 這個方法override是一個open的方法,則也需要表明訪問權(quán)限
override func bar() { }
}
open class SubclassD : SubclassableParentClass {
// 正確的寫法币绩,方法也需要標(biāo)記為open
open override func bar() { }
}
open class SubclassE : SubclassableParentClass {
// 也可以顯式的指出這個方法不能在被override
public final override func bar() { }
}