构造方法

作者: 西门丨不吹雪 | 来源:发表于2016-09-05 17:51 被阅读29次

构造方法:

  • 作用:
    对实例对象的内容进行初始化
    Swift要求类或者结构体中的存储属性(非lazy的)在对象构造完毕后要有初始化值

  • 语法:
    init(参数列表){ 初始化代码 }

  • 注意:
    1.在Swift中类/结构体/枚举都需要构造方法
    2.构造方法的作用仅仅是用于初始化属性, 而不是分配内存, 分配内存是系统帮我们做的
    3.构造方法是隐式调用的, 通过 类名称() 形式创建一个对象就会隐式调用init()构造方法
    4.如果所有的存储属性都有默认值, 可以不提供构造方法, 系统会提供一个隐式的构造方法
    5.如果存储属性可以提供缺省, 那么提倡大家使用设置缺省值的方式, 这样可以简化代码(不用自定义构造方法, 不用写存储属性类型)

class Person {
var name:String = "lnj"
//    var age:Int = 30
var age:Int
func description() ->String{
    return "name = \(name) age = \(age)"
}
init()
{
    print("init")
    age = 30
}
}
// 1.分配内存 2.初始化name和age 3.构造方法是隐式调用的
var p = Person()
p.description() //显示调用

带参数的构造方法

class Person2 {
var name:String
var age:Int
func description() ->String{
    return "name = \(name) age = \(age)"
}
// 构造方法的内部参数, 默认也是外部参数
// 而函数的内部参数默认不会当做外部参数
// 而方法的内部参数, 从第二个开始才会当做外部参数
//    init(name:String, age:Int)
// 构造方法对属性的顺序没有要求, 只要保证对象构造完时所有存储属性被初始化即可
init(age:Int, name:String)
{
    self.name = name
    self.age = age
}
func setName(name:String, age:Int)
{   
}
}
//var p2 = Person(name: "lnj", age: 30)
var p2 = Person2(age: 30, name: "lnj")
p2.setName("lnj", age: 30)
func setName(name:String, age:Int)
{
}
p2.setName("lnj", age: 30)

常量存储属性与构造方法
常量存储属性只能通过缺省值或在构造方法中被修改, 其它任何地方都不能修改

class Person3 {
var name:String = "lnj"
var age:Int
init(age:Int, name:String)
{
    self.age = age
    self.name = name
}
func description() ->String{
    return "name = \(name) age = \(age)"
}
}
var p3 = Person3(age: 30, name:"zs")
print(p3.description())
p3.age = 55
//p3.name = "xxx" // 常量存储属性初始化之后不允许被修改

可选属性与构造方法

class Car{
    let name:String
    init(name:String){
    self.name = name
    }
}
class Person4 {
    let name:String
    var age:Int
    var car:Car?
// 可选值存储属性可以不再构造方法中初始化, 
// 也就是说可选值在对象构造完毕后不用初始化
// 其实如果不对可选存储属性进行初始化, 默认就是nil
init(age:Int, name:String)
{
    self.age = age
    self.name = name
}
func description() ->String{
    return "name = \(name) age = \(age)"
}
}

结构体构造方法

struct Rect{
// 此事即没有提供缺省值, 也没有提供构造方法, 但是编译通过
// 因为默认情况下, 结构体会给结构体提供一个默认的成员逐一构造器
var width:Double = 0.0
var heigth:Double = 0.0
// 系统默认会提供一个类似的方法
init(width:Double, height:Double)
{
    self.width = width
    self.heigth = heigth
}
init()
{
    self.width = 0.0
    self.heigth = 0.0
}
}
// 注意: 1.在类中默认是没有逐一构造器的
// 2.如果在结构体中自定义了构造方法, 那么系统不会生成默认的逐一构造器
// 3.如果给存储属性提供了缺省值, 系统还是会提供默认的逐一构造器
//var r = Rect(width: 1.0, heigth: 1.0)
// 4.如果给存储属性提供了缺省值, 可以使用不带参数的方法初始化结构体
var r = Rect()

"值类型"的构造器代理

  • 构造器代理: 构造方法之间的相互调用
    构造方法可以调用其他构造方法来完成实例的构造, 称之为构造器代理
    好处: 减少构造方法之间的重复代码
struct Rect2 {
    var width:Double
    var height:Double
init(width:Double, height:Double)
{
    self.width = width
    self.height = height
}
init()
{
//        self.width = 0
//        self.height = 0
    // 构造器代理
    self.init(width:0, height:0)
}
func show(){
    print("width = \(width) height = \(height)")
}
}
var r2 = Rect2()
r2.show()
var r3 = Rect2(width: 100, height: 100)
r3.show()

通过闭包或者全局函数/类方法 设置存储属性的缺省值
如果需要经过计算, 或者需要进行一些额外的操作才能确定初始值时就可以通过闭包或全局函数设置存储属性的缺省值

func getValue() ->Int
{
   print("getValue")
   return 55
}
class Person5 {
var name: String
// 系统在初始化的时候会隐式执行闭包, 将闭包的执行结果赋值给存储属性
// 注意: 闭包后面一定要有(), 代表执行闭包
var age: Int = {
//        () -> Int in // 返回值可以省略, 默认返回值的类型就是存储属性的类型
    print("age 闭包")
    return 30
}()
lazy var height:Double = {
    print("lazy 闭包")
   return 175.0
}()

var age2:Int = getValue()
var age3:Int = Person5.getValue2()
// 不能这样写, 因为调用方法时对象还没有初始化完毕
// self只有当所有的存储属性都初始化完毕之后才可以用
//    var age3:Int = self.getValue3()
init(name:String)
{
    print("init")
    self.name = name
}
class func getValue2() -> Int{
    print("class getValue2")
    return 100
}
func getValue3() -> Int
{
    return 88
}
}
var p5 = Person5(name: "lnj")
// 懒加载是用到时才执行, 而闭包赋值是初始化时就会执行
print(p5.height)

继承与构造方法:
指定构造与便利构造方法

class Person {
    var name:String
    var age:Int
// 指定构造方法都是以init开头的
init(name:String, age:Int)
{
    self.name = name
    self.age = age
}
// 如果是值类型没问题, 称之为构造器代理
// 但如果是引用类型会报错, 需要在前面加上convenience关键字
// 被convenience关键字修饰的构造方法称之为便利构造器, 通过调用其它构造方法来初始化
// 反而言之, 便利构造器中一定是调用其它构造方法初始化的, 一定要出现self.init
convenience init()
{
  
//        self.name = "lnj"
//        self.age = 30
    self.init(name:"lnj", age:30)
}
// 类可以拥有多个构造方法
init(name:String)
{
    self.name = name
    self.age = 0
    // 不能在指定构造方法中调用便利构造方法
    // 也就是说指定构造方法中不能出现self.init
//        self.init()
}

convenience init(age:Int)
{
    // 可以在便利构造器中调用指定构造器
//        self.init(name:"lnj", age:30)
    // 可以在便利构造器中调用便利构造器
    self.init()
}
// 便利构造器不能和指定构造器同名
//    convenience init(name:String)
//    {
//    }
}

派生类的构造方法

class Man {
var name:String
// 指定构造方法
init(name:String){
    self.name = name
}
// 便利构造方法
convenience init(){
    self.init(name:"lnj")
}
}
class SuperMan: Man {
var age: Int
注意: 
1.默认情况下构造方法不会被继承
2.基类的存储属性只能通过基类的构造方法初始化
3.初始化存储属性时必须先初始化当前类再初始化父类
4.不能通过便利构造方法初始化父类, 只能通过调用指定构造方法初始化父类
// 指定构造器
init(age:Int){
    self.age = age
    super.init(name: "lnj")
//        super.init()
}
}
  • 构造器间的调用规则
    指定构造器必须调用其直接父类的"指定构造器"
    便利构造器必须调用同类中的其它构造器(指定或便利)
    便利构造器必须最终以调用一个指定构造器结束(无论调用的时指定还是便利, 最终肯定会调用一个指定构造器)
    指定构造器总是向上代理(父类)
    便利构造器总是横向代理(当前类)
class Man2 {
var name:String
// 指定构造方法
init(name:String){
    self.name = name
}
// 便利构造方法
convenience init(){
    self.init(name:"lnj")
}
}
class SuperMan2: Man2 {
var age: Int
// 指定构造器
init(age:Int){
    self.age = age
    super.init(name: "lnj")
}
convenience init(){
    self.init(age:30)
}
convenience init(name:String, age:Int){
调用子类构造器一定能够初始化所有属性
self.init(age:30)
便利构造器中只能通过self.init来初始化, 不能使用super.init
因为调用父类构造器不一定能完全初始化所有属性(子类特有)
 super.init(name: "lnj")
        self.init()
}
}
  • 两段式构造- 构造过程可以划分为两个阶段
    1.确保当前类和父类所有存储属性都被初始化
    2.做一些其它初始化操作

  • 好处:
    1.可以防止属性在被初始化之前访问,
    2.可以防止属性被另外一个构造器意外赋值
    注意:构造器的初始化永远是在所有类的第一阶段初始化完毕之后才会开始第二阶段

  • 编译器安全检查:
    1.必须先初始化子类特有属性, 再向上代理父类指定构造方法初始化父类属性
    2.只能在调用完父类指定构造器后才能访问父类属性
    3.在遍历构造器中, 必须先调用同类其它构造方法后才能访问属性
    4.第一阶段完成前不能访问父类属性/也不能引用self和调用任何实例方法

class Man3 {
var name:String
// 指定构造方法
init(name:String){
    self.name = name
}
// 便利构造方法
convenience init(){
    self.init(name:"lnj")
}
}
class SuperMan3: Man3 {
var age: Int
// 指定构造器
init(age:Int){
    print("SuperMan第一阶段开始")
    // 对子类引入的属性初始化
    self.age = age
    // 代码会报错,因为调用self.name之前还没有对父类的name进行初始化
    // 即便在这个地方修改, 也会被后面的初始化语句覆盖
//        if (age > 30){
//            self.name = "zs"
//        }
    // 对父类引入的属性进行初始化
    super.init(name: "lnj")
    
    print("SuperMan第二阶段开始")
    if age > 30 {
        self.name = "zs"
    }
}
}
class MonkeyMan:SuperMan3{
var height:Double
init(height:Double){
    print("MonkeyMan第一阶段开始")
    // 对子类引入的属性初始化
    self.height = 99.0
    // 对父类引入的属性进行初始化
    super.init(age: 30)
    
    print("MonkeyMan第二阶段开始")
    if height < 100.0{
        self.age = 50
    }
}
}
var m = MonkeyMan(height: 20)

重写指定构造方法: 子类的构造方法和父类的一模一样

class Man4 {
var name:String
// 指定构造方法
init(name:String){
    self.name = name
}
}
class SuperMan4: Man4 {
var age: Int
init(){
    self.age = 30
    super.init(name: "lnj")
}
// 如果子类中的构造器和父类一模一样, 必须加上override关键字, 表示重写父类方法
// 在老版本的Swift语言中是不需要override关键字的, 新版本才推出的
//    override init(name:String){
//        self.age = 30
//        super.init(name: name)
//    }

// 将父类的指定构造器重写成一个便利构造器, 也必须加上override关键字, 表示重写父类方法
convenience override init(name:String){
    self.init(name:name)
    self.age = 30
}
}

便利构造方法不存在重写

class Man5 {
var name:String
// 指定构造方法
init(name:String){
    self.name = name
}
convenience init(){
    self.init(name:"lnj")
}
}
class SuperMan5: Man5 {
var age: Int
init(age:Int){
    self.age = age
    super.init(name: "lnj")
    }
// Swift中便利构造方法不存在重写, 如果加上override关键字, 系统会去查找父类中有没有和便利构造方法一样的指定构造方法, 有旧不报错, 没有就报错
// 为什么便利构造器不能重写呢? 因为便利构造器只能横向代理, 只能调用当前类的其它构造方法或指定方法, 不可能调用super. 所以不存在重写
// 也就是说子类的便利构造方法不能直接访问父类的便利构造方法, 所以不存在重写的概念
convenience init(){
    self.init(age:30)
}
}
// 早期版本中如果字符类中有同名便利构造器会报错
var sm = SuperMan5()

构造方法的自动继承

如果子类中没有定义任何构造器, 且子类中所有的存储属性都有缺省值, 会继承父类中所有的构造方法(包括便利构造器)
如果子类中只是重写了父类中的某些指定构造器, 不管子类中的存储属性是否有缺省值, 都不会继承父类中的其它构造方法
如果子类重写了父类中所有的指定构造器, 不管子类中的存储属性是否有缺省值, 都会同时继承父类中的所有便利方法

class Person6 {
var name:String
var age:Int
init(name:String, age:Int){
    self.name = name
    self.age = age
}
init(name:String){
    self.name = name
    self.age = 0
}
convenience init(){
    self.init(name:"lnj")
}
}
class SuperMan6: Person6 {
//    var height:Double = 175.0
var height:Double

init(height:Double){
    self.height = height
    super.init(name: "lnj", age: 30)
}

override init(name:String, age:Int){
   self.height = 175.0
   super.init(name: name, age: age)
}
override init(name:String){
    self.height = 175.0
    super.init(name: name)
}
}
// 如果子类中没有定义任何构造器, 且子类中所有的存储属性都有缺省值, 会继承父类中所有的构造方法(包括便利构造器)
// 父类的存储属性是由父类的构造器初始化, 子类的存储属性是由缺省值初始化的
//var sm = SuperMan(name: "lnj", age: 30)
//var sm = SuperMan(name: "zs")
//var sm = SuperMan()
//print(sm.height)

// 如果子类中只是重写了父类中的某些指定构造器, 不管子类中的存储属性是否有缺省值, 都不会继承父类中的其它构造方法
//var sm = SuperMan(height: 188.0)

// 如果子类重写了父类中所有的指定构造器, 不管子类中的存储属性是否有缺省值, 都会同时继承父类中的所有便利方法
var sm6 = SuperMan6()

必须构造器:
只要在构造方法的前面加上一个required关键字, 那么所有的子类(后续子类)只要定义了构造方法都必须实现该构造方法

class Person7 {
var name:String
// 早期Swift版本中没有这个语法
required init(name:String){
    self.name = name
}
}
class SuperMan7: Person7 {
var age:Int = 30
// 如果子类没有定义构造器, 可以不用重写
init(){
    self.age = 30
    super.init(name: "lnj")
}
// 1.如果子类自定义了构造器, 就必须重写"必须构造器"
// 因为如果子类没有自定义任何构造器, 默认会继承父类构造器, 所以不用重写
// 2.重写必须构造器不用加override关键字, 因为系统看到required关键字就会自动查看父类
// 为什么重写了还需要加上required关键字, 因为所有后续子类都必须重写
required init(name: String) {
    self.age = 30
    super.init(name:name)
}
}
var sm7 = SuperMan7(name: "lnj")

相关文章

  • scala:构造方法与apply

    scala中分主构造方法、辅助构造方法,辅助构造方法必须调用主构造方法 构造方法参数作用域 如果父类有主构造方法,...

  • 8.构造方法与析构方法

    指定构造方法,便利构造方法,构造方法嵌套convenience构造方法继承,构造方法安全性检查4原则,可失败构造方...

  • 构造方法

    构造方法A.指定构造方法;B.便利构造方法; 关系:A.子类的指定构造方法必须调用父类的构造方法;B.便利构造方法...

  • 构造方法、封装、关键字(this、static)和代码块的介绍

    1.构造方法 1.1 构造方法与成员方法的区别 构造方法分为无参构造和有参构造,其中有参构造方法和无参构造方法为方...

  • Swift 引发的一场知识暴动

    一、构造方法 在Swift中,主要有三种构造方法:系统构造方法、普通构造方法与便利构造方法。构造方法一般都是以 i...

  • Mybatis SqlSessionTemplate 实例化过程

    构造方法 SqlSessionTemplate的构造方法源代码如下,其他构造方法,都是在此构造方法上进行的重载。 ...

  • 十四、构造方法

    一、构造方法格式与特点  构造方法的格式:修饰符 构造方法名(参数列表){} 构造方法的体现: ...

  • Kotlin使用技巧

    Kotlin的构造方法分为主构造方法(primary)和次构造方法(secondary),主构造方法定义在类名之后...

  • 构造方法

    构造方法的概念 构造方法(constructor),有的地方叫做构造器或者构造函数。构造方法的作用是给对象数据进行...

  • Swift-9.构造方法初印象

    本章包含内容: 构造方法初步 常量属性和构造方法 可选属性与构造方法 结构体的构造方法 值类型的构造器代理 闭包或...

网友评论

    本文标题:构造方法

    本文链接:https://www.haomeiwen.com/subject/hrwrettx.html