Swift 结构体构造器

2021-04-09 14:49:09 浏览数 (1)

  • Swift 中的结构体和类跟其它面向对象语言一样都有构造函数, 而OC是没有的
  • Swift 要求实例化一个结构体或类的时候, 所有的成员变量都必须有初始值
  • 构造函数的意义就是用于初始化所有成员变量的, 而不是分配内存, 分配内存是系统帮我们做的.
  • 如果结构体中的所有属性都有默认值, 可以调用 ()构造一个结构体实例如果结构体中的属性没有默认值, 可以自定义构造器, 并在构造器中给所有的属性赋值

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

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

代码语言:javascript复制
struct User {
    var name: String
    var age: Int
}

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

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

代码语言:javascript复制
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即可创建实例:

代码语言:javascript复制
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私有属性,只要这些属性具有默认值,我们还是可以正常使用其成员构造器,和上面没有差异,但是如果私有属性没有默认值,则必须手动编写该类型的构造器——以便能够从外部传入值为该属性赋值:

代码语言:javascript复制
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 中声明该便利构造器,这样做的好处是,当我们定义一些便利构造器方便初始化的同时,不会覆盖编译器生成的成员构造器:

代码语言:javascript复制
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代码会报错,只剩下新声明的便利构造器。

0 人点赞