构造方法总结二

构造器间的调用规则

指定构造器必须调用其直接父类的"指定构造器"

便利构造器必须调用同类中的其它构造器(指定或便利)

便利构造器必须最终以调用一个指定构造器结束(无论调用的时指定还是便利, 最终肯定会调用一个指定构造器)

指定构造器总是向上代理(父类)

便利构造器总是横向代理(当前类)

指定构造与便利构造方法


class Person{
    
    var name : String
    var age : Int
    
    /*
     指定构造方法都是以init开头的
     */
    init(name:String,age:Int) {
        
        self.name = name
        self.age = age
    }
    
    /*
     被convenience关键字修饰的构造方法称之为便利构造器, 通过调用其它构造方法来初始化
     反而言之, 便利构造器中一定是调用其它构造方法初始化的, 一定要出现self.init
    */
    convenience init() {
        
        self.init(name: "zx", age: 30)
    }
    
    /* 
     类可以拥有多个构造方法
     不能在指定构造方法中调用便利构造方法
     也就是说指定构造方法中不能出现self.init
     */
    init(name:String) {
        
        self.name = name
        self.age = 0
    }
    
   
    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: "zx")
    }
}


/*
 注意:
 1.默认情况下构造方法不会被继承
 2.基类的存储属性只能通过基类的构造方法初始化
 3.初始化存储属性时必须先初始化当前类再初始化父类
 4.不能通过便利构造方法初始化父类, 只能通过调用指定构造方法初始化父类
 */
class SuperMan:Man{
    var age : Int
    // 指定构造器
    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:"zx")
    }
}


class SuperMan2: Man2 {
    var age: Int
    // 指定构造器
    init(age:Int){
        self.age = age
        super.init(name: "zx")
    }
    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. 只能在调用完父类指定构造器后才能访问父类属性
  3. 在遍历构造器中,必须先调用同类其它构造方法后才能访问属性
  4. 第一阶段完成前不能访问父类属性/也不能引用self和调用任何实例方法
class Man3 {
    var name:String
    // 指定构造方法
    init(name:String){
        self.name = name
    }
    // 便利构造方法
    convenience init(){
        self.init(name:"zx")
    }
}


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)
m.name



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



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关键字, 表示重写父类方法
//    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")

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容