参考了 没故事的卓同学 的文章:Swift 3必看:新的访问控制fileprivate和open , 替换了一个更为直观的例子
open
- open 修饰的 class 在 Module 内部和外部都可以被访问和继承
- open 修饰的 func 在 Module 内部和外部都可以被访问和重载(override)
Public
- public 修饰的 class 在 Module 内部可以访问和继承,在外部只能访问
- public 修饰的 func 在 Module 内部可以被访问和重载(override),在外部只能访问
Final
- final 修饰的 class 任何地方都不能不能被继承
- final 修饰的 func 任何地方都不能被 Override
总结:
现在的访问权限则依次为:open,public,internal,fileprivate,private。
下面是例子:
- ModuleA.framework 中新建一个类ModuleA.swift
import Foundation
/// final的含义保持不变
public final class FinalClass { }
// 这个类在ModuleA的范围外是不能被继承的,只能被访问
public class PublicClass {
public func testPublic() {}
// 这是错误的写法,因为class已经不能被继承,
// 所以他的方法的访问权限不能大于类的访问权限
open func testOpen() {}
// final的含义保持不变
public final func testPublicFinal() {}
}
// 在ModuleA的范围外可以被继承
open class OpenClass {
// 这个属性在ModuleA的范围外不能被override
public var size : Int = 0
// 这个方法在ModuleA的范围外不能被override
public func testPublic() {}
// 这个方法在任何地方都可以被override
open func testOpen() {}
///final的含义保持不变
public final func testPublicFinal() {}
}
- 在 ModuleB.framework 中新建一个类:ModuleB.swift
- 并且把 ModuleA.framework import 进来
import Foundation
import ModuleA
// 这个写法是错误的,编译会失败,类访问权限标记的是public,只能被访问不能被继承
class SubA : PublicClass { }
// 这样写法可以通过,Class访问权限为 `open`.
class SubB : OpenClass {
// 这样写也会编译失败,因为这个方法权限为public,不是`open'.
override func testPublic() { }
// 这个方法因为在SubclassableParentClass中标记为open,所以可以这样写
// 这里不需要再声明为open,因为这个类是internal的
override func testOpen() { }
}
open class SubC : OpenClass {
// 这种写法会编译失败
override func testPublicFinal() { }
// 正确的写法,方法也需要标记为open
open override func testOpen() { }
}
open class SubE : OpenClass {
// 也可以显式的指出这个方法不能再被override
public final override func testOpen() { }
}