Swift 3.0 的 open,public,internal,fileprivate,private 关键字

参考了 没故事的卓同学 的文章: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() { }
}

推荐阅读更多精彩内容