访问控制(Access Control)
在访问权限控制这块,Swift提供了5个不同的访问级别(以下是从高到低排列, 实体指被访问级别修饰的内容)
open:允许在定义实体的模块、其他模块中访问,允许其他模块进行继承、重写(open只能用在类、类成员上) ;
public:允许在定义实体的模块、其他模块中访问,不允许其他模块进行继承、重写;
internal:只允许在定义实体的模块中访问,不允许在其他模块中访问;
fileprivate:只允许在定义实体的源文件中访问;
private:只允许在定义实体的封闭声明中访问 n绝大部分实体默认都是internal 级别;
注:绝大部分实体默认都是internal级别的
访问级别使用准则
一个实体不可以被更低访问级别的实体定义,比如:
1、变量\常量类型的访问级别 ≥ 变量\常量访问级别
2、参数类型、返回值类型 ≥ 函数
3、父类 ≥ 子类
4、父协议 ≥ 子协议
5、原类型 ≥ typealias
6、原始值类型、关联值类型 ≥ 枚举类型
7、定义类型A时用到的其他类型 ≥ 类型A
8、元组类型的访问级别是所有成员类型最低的那个
9、泛型类型的访问级别是 类型的访问级别 以及 所有泛型类型参数的访问级别 中最低的那个
10、类型的访问级别会影响成员(属性、方法、初始化器、下标)、嵌套类型的默认访问级别;对于类,一般成员变量的访问作用域和类的作用域一致,除非有单独的设置。
11、getter、setter默认自动接收它们所属环境的访问级别
可以给setter单独设置一个比getter更低的访问级别,用以限制写的权限
fileprivate(set) public var num = 10
class Person {
private(set) var age = 0
fileprivate(set) public var weight: Int {
set {}
get { 10 }
}
internal(set) public subscript(index: Int) -> Int {
set {}
get { index }
}
}
12、不能给enum的每个case单独设置访问级别,每个case自动接收enum的访问级别 ,public enum定义的case也是public;
13、协议中定义的要求自动接收协议的访问级别,不能单独设置访问级别。public协议定义的要求也是public。协议实现的访问级别必须 ≥ 类型的访问级别,或者 ≥ 协议的访问级别
public protocol Runnable {
func run()
} // 协议的访问级别是public,则要求实现的访问级别大于或等于public
public class Person : Runnable {
func run() {}
}
14、 在同一文件中的扩展,可以写成类似多个部分的类型声明;在原本的声明中声明一个私有成员,可以在同一文件的扩展中访问它;在扩展中声明一个私有成员,可以在同一文件的其他扩展中、原本声明中访问它。
// 写在同一个文件中,同一个类的多个拓展,虽然run0 函数是private,但其他拓展中依然可以访问(注意必须是写在同一个文件中才行)。这种情况相当于拓展中的代码写在了类里。
public class Person {
private func run0() {}
private func eat0() {
run1()
}
}
extension Person {
private func run1() {}
private func eat1() {
run0()
}
}
extension Person {
private func eat2() {
run1()
}
}