SwiftUI中五种访问修饰符

在SwiftUI中,访问修饰符有5种,分别是:private、fileprivate、internal、public、open。其中fileprivate和open是swift 3 新添加的。由于之前的访问控制符是基于文件的,不是基于类的。这样会有问题,故swift 3 增加了两个修饰符,对原来的private、public进行了细分。

从高到低的排序如下:

open > public > interal > fileprivate > private

(1)private:修饰的属性或者方法只能在当前类中访问。

(2)fileprivate:修饰的属性或者方法只能在当前文件中访问。如果一个文件中含有多个类,也是可以访问的。
(3)internal:默认访问级别,可写可不写。

修饰的属性和方法在源代码的整个模块都可以访问。

如果是框架或库代码,则在整个框架都可以访问,框架以外是不可以访问的。

如果是App代码,则在整个App内部都是可以访问的。

(4)public:可以被任何代码访问。但其他模块不可以被override和继承,而在模块内是可以被override和继承的。

(5)open:可以被任何模块的代码访问,包括override和继承。

private和fileprivate区别

这两种访问控制形式相似,但是特别有一定区别区别。我们使用private可以限制内容可见性。下面就是显示demo:

class CancelableTimer: NSObject {
    private var q = DispatchQueue(label: "timer")
    private var timer : DispatchSourceTimer!
    private var firsttime = true
    private var once : Bool
    private var handler : () -> ()
    init(once:Bool, handler:@escaping () -> ()) {
        // ...
    }
    func start(withInterval interval:Double) {
        // ...
    }
    func cancel() {
        // ...
    }
}

private在父类、嵌套类和extension中的作用情况

(1)private严格保护类的隐私,甚至子类也看不到其超类的私有成员。有时我们需要让某些成员或类只能在一个文件中可见,我们可以用fileprivate
(2)嵌套类型可以查看嵌套类型的私有成员。这是有道理的,因为外部类型是一个范围。嵌套类型可以看到此类型内其他所有内容。
(3)一个extension可以看到扩展类型的private内容。

class Dog {
    private var whatADogSays = "woof"
}
extension Dog {
    func speak() {
        print(self.whatADogSays) // ok
    }
}

private限制set和get

在某些情况下,您可能希望分别设置变量set和get的privacy。为了实现这个需,swift中单词放在其自己的隐私声明之后的括号中,例如 private(set)var myVar表示此变量的set是受限制的,但是对于获取此变量没有限制。同样,您可以说public private(set)var myVar使此变量公开,而将此变量set为private。

fileprivate 只作用于文件中

import SwiftUI

struct Picker_RowsWithImages : View {
    @State private var youTuberName = "水果"
    
    var body: some View {
        VStack(spacing: 20) {
            Text("带图片的Picker").font(.largeTitle)
            Text("图片显示").font(.title).foregroundColor(.gray)
            Text("今天您想吃什么水果?")
                .padding(.top)
            
            Picker(selection: $youTuberName, label: Text("")) {
                Row(name: "苹果")
                Row(name: "香蕉")
                Row(name: "芒果")
                Row(name: "西瓜")
                Row(name: "菠萝")
            }
            .foregroundColor(Color.white)
            .padding(.horizontal)
            .background(Color.orange)
            .cornerRadius(15)
            .shadow(radius: 20)
            .labelsHidden()
        }
    }
}

fileprivate struct Row : View {
    var name: String
    
    var body: some View {
        return HStack {
            Image(systemName: "tray.fill")
                .padding(.trailing)
                .foregroundColor(Color.red)
            Text(name)
        }
        .tag(name)
    }
}

代码解析,Row只能在file中使用,而不能被项目其他file引用。