IOS

Swift 结构体构造器

2021-04-03  本文已影响0人  韦弦Zhy

在Swift中,定义为结构体的类型会自动获得由编译器生成的默认初始化程序——所谓的“成员构造器”,因为编译器将根据给定结构体的成员(即其存储的属性)生成该初始化程序。

例如,如果我们定义了一个User具有nameage属性的结构体,则可以使用其成员构造器来简单地通过传递这两个属性的值来创建实例:

struct User {
    var name: String
    var age: Int
}

let user = User(name: "韦弦", age: 9)

另一方面,当编译器合成成员构造器时,将完全忽略计算属性——因此,即使我们添加一个成员属性,我们仍然可以像以前一样继续使用上述初始化程序:

struct User {
    var name: String
    var age: Int
    var isAdult: Bool { age >= 18 }
}

let user = User(name: "韦弦", age: 9)

从 Swift 5.1 开始,成员构造器也考虑了默认属性值——这意味着,如果我们为age属性提供默认值,则User只需传递一个name即可创建实例:

struct User {
    var name: String
    var age: Int = 18
    var isAdult: Bool { age >= 18 }
}

// 有如下两种初始化方式
let user1 = User(name: "zhy")
let user2 = User(name: "韦弦", age: 9)

如果该类型具有private私有属性,只要这些属性具有默认值,我们还是可以正常使用其成员构造器,和上面没有差异,但是如果私有属性没有默认值,则必须手动编写该类型的构造器——以便能够从外部传入值为该属性赋值:

struct User {
    var name: String
    private var age: Int
    var isAdult: Bool { age >= 18 }

    init(name: String, age: Int) {
        self.name = name
        self.age = age
    }
}

let user = User(name: "韦弦", age: 9)

不过,要记住的一件事是,成员构造器永远不会具有高于internal的访问级别,这意味着我们只能在定义其类型的模块内部使用它们。

最初,这似乎是一个奇怪的限制,但它确实有其优点,因为我们可以说应该始终设计供公众使用的显式API,而不必将它们与数据的内部结构联系在一起。

因此,总而言之,在以下情况下,我们可以使用结构体默认生成的成员构造器:

所有其他情况都要求我们至少到目前为止至少手动实现一个初始化程序。

当我们为结构体创建便利构造器的时候,我们可以在 扩展extension 中声明该便利构造器,这样做的好处是,当我们定义一些便利构造器方便初始化的同时,不会覆盖编译器生成的成员构造器:

struct User {
    var name: String
    var gender: Gender
    var age: Int
    var isAdult: Bool { age >= 18 }

    enum Gender {
        case man,woman,other
    }
}


extension User {
    init(name: String, age: Int) {
        self.init(name: name, gender: .man, age: 18)
    }
}

// 可以有如下两种初始化方法
let user1 = User(name: "韦弦", age: 9)
let user2 = User(name: "zhy", gender: .man, age: 18)

如果直接写在结构体内,则初始化 user2代码会报错,只剩下新声明的便利构造器。

上一篇下一篇

猜你喜欢

热点阅读