swift编程开发

Swift:便利构造器

2016-05-09  本文已影响1584人  厨子

由一段对象序列化的代码,分析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 ,这个要求也是满足的。可参看下面两条规则:

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

两种构造器之间的使用规则
规则图一
上面的 Subclass 满足上面的三条规则, Superclass没有父类,所以不应用第一条规则。再看一个复杂的图例: 规则图二
初始化之两步走

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

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

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

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

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

上一篇下一篇

猜你喜欢

热点阅读