官网地址:https://developer.apple.com/library/content/documentation/Swift/Conceptual/Swift_Programming_Language/AccessControl.html#//apple_ref/doc/uid/TP40014097-CH41-ID3
1、访问控制种类
swift3.0以后,访问控制主要分为五类:open、public、internal、fileprivate、private,其中open级别最高,private最低。
访问控制 | 定义 |
open | 可以访问自己模块中源文件里面的任何实体,其他模块也可以通过引入该模块中源文件访问所有的实体,并且其他模块可以继承本模块的实体和重新本模块中实体的方法 |
public | 可以访问自己模块中源文件里面的任何实体,其他模块也可以通过引入该模块中源文件访问所有的实体,但其他模块不可以继承本模块的实体和重新本模块中实体的方法 |
internal | 可以被自己模块中代码访问其他源文件里的任何实体,但是其他模块不能访问该模块中源文件里面的实体 |
fileprivate | 文件内私有,只能在当前源文件使用 |
private | 只能在类中访问,离开了这个类或者结构体的作用域以后就无法访问 |
注:
模块:以独立单元构建和发布的framework或app。通过XCODE bulid出来的target就可以看做是一个独立模块。
源文件:一个模块里面的一个swift代码文件,一个源文件里面可以包含多个类、结构等。
2、语法
总法则有两条:
1)不能给一个变量设置高于其类型访问权限的访问权限,比如
2)函数的访问权限由函数参数和返回值的权限决定,取两者权限的最小值。
默认权限
如果没有给一个实体指定访问权限,其默认权限为internal
权限控制的语法
public class SomePublicClass {}
internal class SomeInternalClass {}
fileprivate class SomeFilePrivateClass {}
private class SomePrivateClass {}
public var somePublicVariable = 0
internal let someInternalConstant = 0
fileprivate func someFilePrivateFunction() {}
private func somePrivateFunction() {}
一个实体的权限控制也影响着该实体成员的权限:如果一个实体权限控制为private或fileprivate,那么他的成员默认访问权限为private或fileprivate;如果一个实体权限控制为internal或public,那么他的成员默认访问权限为internal。如果想让一个访问权限为public的实体的成员权限也为public,必须自己指定该成员的权限为public。
public class SomePublicClass { // explicitly public class
public var somePublicProperty = 0 // explicitly public class member
var someInternalProperty = 0 // implicitly internal class member
fileprivate func someFilePrivateMethod() {} // explicitly file-private class member
private func somePrivateMethod() {} // explicitly private class member
}
class SomeInternalClass { // implicitly internal class
var someInternalProperty = 0 // implicitly internal class member
fileprivate func someFilePrivateMethod() {} // explicitly file-private class member
private func somePrivateMethod() {} // explicitly private class member
}
fileprivate class SomeFilePrivateClass { // explicitly file-private class
func someFilePrivateMethod() {} // implicitly file-private class member
private func somePrivateMethod() {} // explicitly private class member
}
private class SomePrivateClass { // explicitly private class
func somePrivateMethod() {} // implicitly private class member
}
元组的访问权限
元组的访问权限取决于他包含所有类型的权限最小值。比如一个元组包含两个类型,一个类型的权限为internal,一个类型的权限为private,那么该元组的访问权限为private。
函数的访问权限
函数的访问权限规则已经在上面说了,这里不再赘述。直接上例子:
枚举类型的访问权限
枚举成员的访问级别继承自该枚举,不能单独为枚举中的成员单独声明不同的访问权限
子类的访问权限
子类的访问权限不能高于父类的访问权限。比如,父类的访问权限为internal,子类的访问权限就不能为public。另外,你可以在子类里面重写父类里面的(方法,属性,构造函数,下表访问器),但是要遵守三个原则(以方法为例说明):1、能在子类中访问到父类里面的方法;2、可以给重写方法设置新的访问权限;3、新赋予的权限需要大于等于父类该方法的访问权限级别。例子如下:
常量、变量、属性、下标的访问权限
他们的访问权限不能高于对应类型的访问权限。
Getter和Setter访问权限
常量、变量、属性、下标索引的Getter和Setter的访问权限继承自他们所属成员的访问级别。你可以把Setter的访问权限设置地低于对应的Getter的权限,这样就能够控制变量、属性、下标索引的读写权限。设置Setter访问权限的方法为把fileprivate(set),private(set),internal(set)放在变量、属性、下标索引的前面
struct TrackedString {
//numberOfEdits的get权限为internal,set权限为private。所以numberOfEdits只能在TrackedString里面修改
private(set) var numberOfEdits = 0
var value: String = "" {
didSet {
numberOfEdits += 1
}
}
}
如果既想对Getter进行修改,也想对Setter进行修改,可以如下设置:
public struct TrackedString {
//numberOfEdits的get权限为public、set权限为private
public private(set) var numberOfEdits = 0
public var value: String = "" {
didSet {
numberOfEdits += 1
}
}
public init() {}
}
构造器和默认构造器的访问权限
对自定义的构造器设置访问权限的时候,不能高于他所属类的访问权限。但是对于必要构造器,他的访问权限必须跟所属类的访问权限相同。
swift为结构体、类提供了一个默认的无参初始化方法,用于给他们的所有属性赋值,但不会给出具体指。默认初始化方法的访问权限与所属类型的访问权限相同,注意当类、结构体的权限为public时,默认构造器的访问权限为internal
协议的访问权限
如果要为一个协议设置访问权限,那么要确保该协议只在设置的访问权限作用域中使用。一个类可以实现一个比自己访问权限低的协议,比如定义一个Public的类,他又实现了一个internal的协议,那么这个类的访问权限为internal(取两者的最小值)。
扩展的访问权限
扩展成员和原始类成员有一致的访问权限,比如你扩展了一个带有public权限的类,那么新加的成员应该有和原始成员一样的默认为internal的访问权限。另外,可以给扩展设置明确的访问权限(例如:private extension),扩展中成员也可以设置自己的访问权限。