Swift:便利构造器

由一段对象序列化的代码,分析designated initializerconvenience initializer

class Meal: NSObject, NSCoding {
    
    // MARK: Properties
    
    var name:   String
    var photo:  UIImage?
    var rating: Int

   // MARK: Types
    
    struct PropertyKey {
        
        static let nameKey   = "name"
        static let photoKey  = "photo"
        static let ratingKey = "rating"
    }
    
    // MARK: Init
    
    init?(name: String, photo: UIImage?, rating: Int) {
        
        self.name   = name
        self.photo  = photo
        self.rating = rating
        
        super.init()
        
        if name.isEmpty || rating < 0 {
            
            return nil
        }
    }
    
    // MARK: NSCoding
    
    func encodeWithCoder(aCoder: NSCoder) {
        
        aCoder.encodeObject(name, forKey: PropertyKey.nameKey)
        aCoder.encodeObject(photo, forKey: PropertyKey.photoKey)
        aCoder.encodeInteger(rating, forKey: PropertyKey.ratingKey)
    }
    required convenience init?(coder aDecoder: NSCoder) {
        
        let name   = aDecoder.decodeObjectForKey(PropertyKey.nameKey) as! String
        let photo  = aDecoder.decodeObjectForKey(PropertyKey.photoKey) as? UIImage
        let rating = aDecoder.decodeIntegerForKey(PropertyKey.ratingKey)
        
        self.init(name: name, photo: photo, rating: rating)
    }
}
Designated initializers

上面代码中,init?(name: String, photo: UIImage?, rating: Int) 是类 Meal 的一个指定构造器(designated initializer)。

designated initializers 是一个类的主构造器,在它们里面可以初始化本类所有已声明的属性,并且可以沿着父类链,向上继续初始化工作。

每一个类必须最少有一个 designated initializer。通过继承父类的一个或多个 designated initializers ,这个要求也是满足的。可参看下面两条规则:

  • 如果子类没有定义任何 designated initializers ,它会自动继承父类所有的 designated initializers
  • 如果子类实现了父类所有的 designated initializers,那么它自动继承父类所有的 convenience initializers
Convenience Initializers

上面代码中,convenience init?(coder aDecoder: NSCoder) 是类 Meal 的一个便利构造器(convenience initializer)。在 init 前加上关键字 convenience 标识。

convenience initializers 是一个类的辅助构造器。你可以定义一个 convenience initializer,这个convenience initializer 带有本类 designated initializer 所需的参数,然后在这个 convenience initializer 里调用本类的 designated initializer,来设置属性的默认值。比如上面的代码,在方法 convenience init?(coder aDecoder: NSCoder) 中完成解码,然后作为参数传递给方法 init?(name: String, photo: UIImage?, rating: Int),完成属性的初始化。

convenience initializers 对于类来说并不是必需的,如果创建 convenience initializers 可以简化初始化模式,或者让初始化变得更加清晰,那么就去创建 convenience initializers

两种构造器之间的使用规则
  • 一个 designated initializer 必须调用它直接父类(如果有父类)的一个 designated initializer
  • 一个 convenience initializer 必须调用本类的一个 designated initializer
  • 一个 convenience initializer 最终调用的必须是一个 designated initializer

规则图一

上面的 Subclass 满足上面的三条规则, Superclass没有父类,所以不应用第一条规则。再看一个复杂的图例:

规则图二
初始化之两步走

Swift 中类的初始化分两步。
第一步,在类中声明的每一个存储属性都被赋予一个初始值;
第二步,第一步结束后,在新实例被使用之前,每一个类(比如有继承关系的多个类)可以自定义它的存储属性。

这两步,不仅使得初始化更加安全,并且在类的层级中,也使各个类更加灵活。

OC 中,第一步会把每一个属性赋值为 0nilSwift 比较灵活,可以自定义初始值,还可以处理默认值不能为 0nil的类型。

Swift 编译器会做以下4个安全性检测,来确保两步初始化正确进行:

  • Safety check 1, 一个 designated initializer 在向上委托给它的父类构造器之前,必须确保它的类所声明的所有属性都已被初始化
  • Safety check 2 一个 designated initializer 在给一个继承的属性赋值之前,先要向上委托给父类的一个构造器,然后再给这个继承的属性赋值。不然,这个新值会被父类构造器重写
  • Safety check 3 一个 convenience initializer 必须委托给另外一个构造器,来给属性赋值。不然,新值会被本类的 designated initializer 重写
  • Safety check 4 在初始化第一步完成之前,构造器不能调用任何一个实例方法,读取实例属性的值,或作为值访问 self

当一个 designatedconvenience initializer 被调用时,便开始了初始化工作。

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

推荐阅读更多精彩内容