Swift中重写重载构造(初始化)方法

构造方法是一个类创建对象最先也是必须调用的方法, 在OC中, 我们更习惯称这类方法为初始化方法. OC中的初始化方法与普通方法相比,并没有太严格的分界除了要以init开头. 在Swift语言体系中,构造方法与普通的方法分界十分严格,从格式写法上就有不同,普通方法函数要以func声明,构造方法统一为init命名,不需要func关键字声明,不同的构造方法采用方法重载方式创建(在OC中并没有重载的概念,因为OC的一个方法每增加一个参数就会变成另外一个新的方法)。

一、Swift方法的重写与重载

重写:

在Swift中,重写父类方法必须需要重写的方法前添加override关键字

  • 定义: 方法重写顾名思义,就是对一个方法的重新实现.
  • 示例
/// 父类中声明一个方法
class SuperClass: NSObject {
    func sayHelloWorld() {
        print("Hello World") //!< 输出 Hello World
    }
}
/// 子类中重写父类的方法
class SubClass: SuperClass {
    override func sayHelloWorld() {
        print("Hello World I am Wang") //!< 输出 Hello World I am Wang
    }
}

(方法)重载:

  • 定义:简单来说就是方法名相同,但是参数列表不同. 这样同名不同参的方法之间互相称之为重载方法.
  • 示例
/// 子类中重载父类的方法
class SubClass: SuperClass {
    /// sayHelloWorld(name: "WangZH")
    func sayHelloWorld(name: String) {
        super.sayHelloWorld()
        print("Hello World I am \(name)") //!< 输出 Hello World I am WangZH
    }
}

二、Swift构造方法的重写和重载

Swift中的构造方法分为Designated构造方法(指定构造方法)与Convenience构造方法(方便构造方法)两类。Designated构造方法不加任何修饰关键字,Convenience构造方法需要使用Convenience关键字进行修饰。简而言之,Convenience构造方法就是为了方便使用Designated构造方法。

NSObject 子类构造方法的重写与重载

  • NSObject子类构造方法的重写、重载和正常的方法并没有什么太大的区别,唯一值得注意的是你要保证,在构造方法完成之前,完成所有成员常量(变量)的构造和赋值,可选值(optional)除外。
  • 以下代码并没有涉及到成员变量(常量)的初始化赋值
class SuperClass: NSObject {
    /// 重写父类init()方法
    override init() {
        super.init()
        print("Say Hello")
    }
}
class Subclass: SuperClass {
    /// 重载父类的init()方法
    init(name: String, word: String) {
        super.init()
        print("\(name) Say \(word)")
    }
    
    /// 创建便利初始化方法,调用了self.init(name: String, word: String)方法
    convenience init(word: String) {
        self.init(name: "WangZH", word: word)
    }
}
  • 在子类的构造方法中改变父类的变量值
class SuperClass: NSObject {
    /// 声明一个number变量
    var number: Int
    override init() {
        /// 初始化完成之前给number赋值
        number = 0
        super.init()
        print("Say Hello")
    }
}
class Subclass: SuperClass {
    /// 声明一个变量numberStr
    var numberStr: String
    init(name: String, word: String) {
        /// 初始化之前完成赋值
        numberStr = "1"
        super.init()
        /// 初始化完成之后修改父类的变量值
        number = 1
        print("\(name) Say \(word)")
    }
}

在子类的构造方法中改变父类的变量值,需要父类初始化完成之后才可以更改,需要注意的是如果在这个构造方法中还需要给自己的变量赋值,则需要放在父类初始化完成之前。**

UIViewController子类构造方法的重写与重载

  • UIViewController子类构造方法的重写、重载和NSObject子类的重写、重载基本上相同,但是有一些需要注意的地方。
    先看以下代码:
class ViewController: UIViewController {
    /// 重写父类的init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?)方法
    override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?) {
        super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil)
    }
    
    /// 重载父类的init()
    init(name: String) {
        super.init(nibName: nil, bundle: nil)
    }
    
    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
    }
}

上面代码重写和重载了父类的构造方法,大家有没有发现重载父类方法的时候super调用的是init(nibName: nibName, bundle: bundle)方法?这是为什么呢? 苹果官方是这么说的:

/*The designated initializer. If you subclass UIViewController, you must call the super implementation of this
method, even if you aren't using a NIB. (As a convenience, the default init method will do this for you,
and specify nil for both of this methods arguments.) In the specified NIB, the File's Owner proxy should
have its class set to your view controller subclass, with the view outlet connected to the main view. If you
invoke this method with a nil nib name, then this class' -loadView method will attempt to load a NIB whose
name is the same as your view controller's class. If no such NIB in fact exists then you must either call
-setView: before -view is invoked, or override the -loadView method to set up your views programatically. */
public init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?)


这段话的大体意思就是VC的`init()`是一个*Convenience*类型的构造方法,它会帮你把`public init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?)`方法的两个参数都设置成nil。也就是说,如果你要重载UIViewController的`init()`方法,就需要手动将上面方法的两个参数置为`nil`。修改后`init()`重载的代码如下:

class ViewController: UIViewController {
/// 便利构造方法
convenience init(name: String) {
/// 根据Swift构造方法的特点,如果子类没有定义任何Designated类型的构造方法,则默认继承父类所有的Designated构造方法。
self.init(nibName: nil, bundle: nil)
}
}

当然,还可以这么写:

class ViewController: UIViewController {
convenience init(name: String) {
/// 其实和上面的原理是一样的,UIViewController的init()方法把init(nibName: nib, bundle: bundle)的两个参数默认设置成了nil, 当然,这样看起来简洁一些
self.init()
}
}


OK,说完重载再回过头来说一下构造方法的重写,看下面代码:

class ViewController: UIViewController {
/// 重写方法
override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?) {
super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil)
}
/// UIViewController中标记子类必须要实现的方法
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}

上面代码重写了UIViewController的`init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?)`方法,编译器没有报错,运行没有问题,那么这个就可以跳过了。等等,下面的`required init?(coder aDecoder: NSCoder)`方法是什么鬼啊?Ok,先来说一下`required`这个修饰符,`required`修饰的方法要求子类**必须实现**,如果子类不实现,编译是不会通过的。那么问题又来了,我们平常用`ViewController`的时候也没有实现这个方法啊,程序跑的也很欢快啊,这是怎么回事呢?因为Swift构造方法的特点之一就是:**"如果子类没有定义任何 **Designated** 类型的构造方法,则默认继承父类所有的 **Designated** 构造方法"**。这下就明了吧,不是子类没有实现,只不过是继承了父类的实现罢了。那么接下来就要说一下UIView的构造方法了。再等等,可是为什么**重写了方法**之后就需要子类来实现了呢?答案还是在Swift构造方法的特点里:**如果子类重写了父类的某一构造方法,则默认不在继承父类所有的构造方法。对于Designated类型的构造方法,子类重写了哪些,哪些才能够被使用。对于Convenienve类型的构造方法,需要在子类的重写方法中调用其依赖的父类Designated构造方法,代码如下:**

class SuperClass: NSObject {
/// 父类convenience方法依赖的designated方法
init(name: String, word: String) {
super.init()
}

convenience init(word: String) {
    self.init(name: "wangzh", word: word)
}

}

class SubClass: SuperClass {
/// 重写父类的convenience方法
init(word: String) {
/// 调用父类convenience依赖的designed方法
super.init(name: "哈哈", word: "ha")

    /// 这种调用是不会被编译通过的
    super.init(word: word)
    
}

func test() {
    let subClass = SubClass.init(word: "哈哈")
    print(subClass)
}

}```

UIView子类构造方法的重写与重载

UIView的构造方法其实就和UIViewController里面的差不多。可以这么说,UIKit里面构造方法的重写和重载都和上面说的UiViewController的差不多,这里就不多赘述了。

感谢:

珲少
tiborbodecs 分享的文章

原创作品,转载请注明出处:http://www.wzh.wiki

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

推荐阅读更多精彩内容

  • 推荐的代码组织方式是将 dealloc 方法放在实现文件的最前面(直接在 @synthesize 以及 @dyna...
    YangPu阅读 672评论 0 4
  • 原文 https://github.com/objc-zen/objc-zen-book译文 https://gi...
    linxiangyu阅读 1,646评论 0 10
  • 官方文档 初始化 Initialization是为准备使用类,结构体或者枚举实例的一个过程。这个过程涉及了在实例里...
    hrscy阅读 1,130评论 0 1
  • 有些时候我们需要用Excel处理大量的数据,这时候如果还用range引用单元格内容就会显得有点慢,这时候如果采用数...
    鸣人吃土豆阅读 22,925评论 1 43
  • 跟弟弟一样,从小便喜欢画画,不是去画那种很厉害的那种,就是喜欢画画动漫人物,或者去临摹一些小说人设。后来高中学业重...
    虎牙27阅读 201评论 2 1