我下面所说的成员,是指的类型的成员变量和成员函数.
一.概述
Swift的权限控制在英语中称为Privacy,又可以叫access control
共有五种:
-
internal
默认权限.
在模块内的文件,其类定义和函数定义是互相可见的,但是模块外是不可见的,所以它又可理解为:对模块私有.
这一点和C,Objective-C不同,在C和Objective-C里面,即使在模块内部,如果没有使用include或者import语句,文件间的对象就彼此不可见. -
fileprivate(比internal权限更小)
只在文件内可见.即:2个定义在同一个文件中的类型互相之间可见,但是其他文件就不可见这两个类型.
-
private(比fileprivate权限更小)
一个声明为private的东东只在定义它的花括号内可见.比如一个类型的private成员变量只在这个类型的声明中可见
但是:一个写在文件顶层的private的声明,它其实是fileprivate的,这个很好理解哈
-
public(比internal权限更大)
在模块外可见.想要这样也不是光声明了public就可以,它需要满足2个条件:
- 另一个模块B必须import这个模块A,
- 模块A想要B可见的内容,如类,类的成员,必须声明为public
这样模块B才可以看到模块A中被声明为public的东西
如果自己实现了一个模块,比如package或则framework,你必须至少把其中的一个类型标记为public,否则别的模块就算import了你的module,也无法看到任何类型. open(比public权限更大)
如果一个模块中的类声明为open,则另一个模块可以子类化这个模块的类,可以重写它的成员
二. 详解
Private 和 Filepivate
-
如果父类的成员声明为private,则子类不能看到父类的private的成员.
如果子类型想要看到父类型的private成员,可以这么做:
- 把父类型的成员为声明从private改成fileprivate
- 把子类型和父类型写在同一个文件中
一个类内嵌的类型可以看到该类所有的私有成员
-
extension可以看到它所扩展的类型的所有私有成员,前提是类型的extension和类型声明写在同一个文件中
例:class Dog { private var whatADogSays = "woof" } extension Dog { func speak() { print(self.whatADogSays) // ok } }
-
一个变量的set和get可以有不同的访问权限
如:private(set) var myVar
意味着该变量的set是private的,但是get还是internal的public private(set) var myVar
意味着该变量的get是public的,set是private的 -
Objective-C 增加了权限访问的复杂度:
- 被标记为@objc的东东是private,但是Objective-C依然可以看见它们.这种情况也包括了被标记为@IBAction和@IBOutlet的对象,因为它们其实是隐式的@objc
- 如果实现了一个Objective-C 的protocol, 那些属于protocol的成员不能被标记成private,
- 并且,在Cocoa定义的protocol中定义的可选实现的方法(optional method)必须至少实现为internal,否则Cocoa找不到也无法调用它们.
Public 和 Open
- public定义了模块中的公有API.一个framework就是一个模块.
- 如果一个公用类型的初始化函数是隐式的,则其他模块并不能创建这个类型的对象,该初始化函数必须被显示声明并标记为public
- 一个public修饰的类.在另一个模块中,这个类只是可见,不能被继承;而open修饰的类可以被继承.
- 一个open修饰的类如果有个public修饰的成员,在另一个模块中,这个成员不能被重写;只有open修饰的成员可以被重写.