Swift中构造方法的解析

Easter79
• 阅读 725

Swift中构造方法的解析

一、引言

      构造方法是一个类创建对象最先也是必须调用的方法,在Objective-C中,开发者更习惯称这类方法为初始化方法。在Objective-C中的初始化方法与普通函数相比除了要以init抬头外并无太严格的分界,而在Swift语言体系中,构造方法与普通的方法分界十分严格,从格式写法上就有不同,普通方法函数要以func声明,构造方法统一为init命名,不需要func关键字声明,不同的构造方法采用方法重载的方式创建。

二、构造方法的复写与重载

     在Objective-C中,不同的初始化方法就是不同的函数,这便不存在方法重载的概念。Swift中要创建自定义的构造方法,需要开发者对init构造方法进行重载操作。任何一个自定义的类,只要其有父类,除了可以继承下来父类已有的构造方法外,还可以复写父类的构造方法,使其适用于自身。和Objective-C类似,复写父类的构造方法时,要在其中调用父类的构造方法,重载可以理解为一种特殊的复写父类构造方法,因此在重载的构造方法中也要调用父类的构造方法。

创建一个继承于NSObject的类,复写构造方法,代码示例如下:

class ClassOne: NSObject {
    //声明一个本类特有的常量
    var tip:Int
    //复写父类的构造方法 需要用override关键字
    override init() {
        //构造方法中要对所有成员常量完成创建
        tip = 1;
        //在创建完所有成员常量后 调用父类构造方法
        super.init()
    }
    //重载构造方法1
    init(one:Int){
        tip=one
        super.init()
    }
    //重载构造方法2 使用convenience关键字进行修饰
    convenience init(two:String) {
        //使用convenience关键字进行修饰的构造方法要调用本类的构造方法进行
        self.init(one: two.characters.count)
    }
    //重载构造方法3 使用required关键字进行修饰 使用required关键字进行修饰的构造方法子类必须继承或复写
    required init(three:Float) {
        tip=10
        super.init()
    }

上面示例代码中,不带参数的init()方法为复写父类的方法,因此需要使用关键字override来修饰。重载构造方法1带一个Int类型的 参数,父类中并没有这个构造方法,但是在其实现中,依然需要调用父类中的某个构造方法完成。构造方法2是一个带String类型参数的构造方法,其用convenience关键字为构造方法的一个修饰关键字,后面会介绍。构造方法3为一个带Float类型参数的构造方法,但其使用required关键字进行了修饰,使用required关键字进行修饰的构造方法子类必须继承或者复写。构造方法1,2,3都是对init()构造方法的一种重载,但却是3中类型全然不同的构造方法。

三、Designated构造方法与Convenience构造方法

      Swift中的构造方法分为Designated构造方法与Convenience构造方法两类,Designated构造方法也被称为指定构造方法,Convenience构造方法也被称为方便构造方法。Designated构造方法不加任何修饰关键字,Convenience构造方法需要使用Convenience关键字进行修饰。可以这样理解,Convenience类型的构造方法是为了方便使用从Designated构造方法中分支出来的构造方法,官方文档中有如下描述:

1.子类Designated构造方法中必须调用父类的Designated构造方法。

2.Convenience构造方法中必须调用当前类的构造方法。

3.Convenience构造方法归根结底要调用到Designated构造方法。

官方文档的一张图可以清晰的描述上述关系:

Swift中构造方法的解析

四、构造方法的继承关系

关于子类继承父类的构造方法有这样几个特性:

1.如果子类没有复写任何父类的构造方法,则默认子类将继承所有父类的构造方法,包括Designated构造方法与Convenience构造方法。

2.如果子类复写了父类某一构造方法,则子类默认不在继承所有父类的构造方法,对于Designated类型的构造方法,子类复写了哪些,哪些才能够被使用,对于Convenienve类型的构造方法,子类复写的其调用的Designated构造方法后会被自动继承。

3.如果父类中的构造方法是required修饰的,则子类必须进行继承或复写。

      曾经有朋友和我抱怨,Objective-C中的继承是一种十分不人性,它强制子类继承所有父类的方法与属性无论子类是否需要,分析上面的一些规则可以发现,Swift与Objective-C相比,在构造方法方面语法会更加严格,这样做在编程上更加安全。在Objective-C中,子类将被强制继承所有父类的初始化方法,这样开发者在使用时常常会出现疑惑,有时一个子类往往有特定的初始化方法,仅仅通过父类的初始化方法不能够正确的完成初始化,在编程时,往往需要特殊注释来提示开发者。Swift设定的这些构造方法原则可以将无关的父类构造方法剔除在外,在编程时更加严格安全,减少疑惑与不可控因素。

五、构造方法的实现原则

      无论Designated类型的构造方法还是Convenience类型的构造方法,只要其有父类,最终都要实现父类的Designated构造方法。Swift语言要求,在构造方法中要完成所有成员常量或者变量的构造或赋值(optional值除外)。在对成员常量或变量进行构造赋值时,要在调用父类的初始化方法之前,这里还有一点需要注意,父类的成员属性也会被子类继承,如果要在子类复写的父类方法中对继承来的父类成员属性进行重新构造或赋值,则必须在调用父类构造方法之后,例如创建ClassTwo类继承于ClassOne,复写方法如下:

class ClassTwo: ClassOne {
    //子类自己的属性
    let tipTwo:Int
    override init() {
        //调用父类构造方法前进行自己属性的构造
        tipTwo = 1
        //调用父类构造方法
        super.init()
        //对从父类继承来的属性进行重构造
        tip = 1000;
    }
    
    required init(three: Float) {
        fatalError("init(three:) has not been implemented")
    }
    

}

Swift语言这种强制化得构造规则,能够保证一个类在完成构造时,其内部的所有属性都构造完成。在使用Objective-C进行开发时,很多初学者都可能会遇到这样一种情况,完成了某个类的初始化,但向类的属性进行赋值时却没有成功,因为Objective-C中并没有这样的语法,在类初始化成功后,其属性是否初始化了完全取决于开发者,Swift优化了这一设计。

    综上可以了解,Swift语言虽然更加严格,却将更多本来需要开发者注意的地方交由了编译器,实际上是减轻了开发者的负担。

专注技术,热爱生活,交流技术,也做朋友。

——珲少 QQ群:203317592

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Wesley13 Wesley13
3年前
java第二次作业
(一)学习总结1.什么是构造方法?什么是构造方法的重载?下面的程序是否可以通过编译?为什么?(1)在对面向对象程序中构造方法的主要作用是为类中的属性初始化。在构造方法中要注意以下几点①构造方法的名称必须与类名称一致②构造方法的声明处不能有任何返回值类型的说明③不能在构造方法中使用return返回一个值(2)构造方法的重载就
九路 九路
3年前
Swift与Objective-C混合编程之Swift与Objective-C API映射
原创文章,欢迎转载。转载请注明:关东升的博客Swift与ObjectiveCAPI映射在混合编程过程中Swift与ObjectiveC调用是双向的,由于不同语言对于相同API的表述是不同的,他们之间是有某种映射规律的,这种API映射规律主要体现在构造函数和方法两个方面。1、构造函数映射在Swift与ObjectiveC语言进行混合
Wesley13 Wesley13
3年前
Java日期时间API系列31
  时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总毫秒数,是所有时间的基础,其他时间可以通过时间戳转换得到。Java中本来已经有相关获取时间戳的方法,Java8后增加新的类Instant等专用于处理时间戳问题。 1获取时间戳的方法和性能对比1.1获取时间戳方法Java8以前
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
3年前
Java构造方法与析构方法实例剖析
Java构造方法类有一个特殊的成员方法叫作构造方法,它的作用是创建对象并初始化成员变量。在创建对象时,会自动调用类的构造方法。构造方法定义规则:Java中的构造方法必须与该类具有相同的名字,并且没有方法的返回类型(包括没有void)。另外,构造方法一般都应用public类型来说明,这样才能在程序任意的位置创建类的实例-
Easter79 Easter79
3年前
Swift讲解专题十二——方法
Swift讲解专题十二——方法一、引言      方法只是一个术语,其实就是将函数与特定的类型结合,类、结构体、枚举都可以定义方法,方法又分为实例方法和类型方法,类型方法类似于ObjectiveC中的类方法。Swift和ObjectiveC的一大不同是,ObjectiveC只有在类中可以定义方法。
Wesley13 Wesley13
3年前
Java日期时间API系列30
  实际使用中,经常需要使用不同精确度的Date,比如保留到天2020042300:00:00,保留到小时,保留到分钟,保留到秒等,常见的方法是通过格式化到指定精确度(比如:yyyyMMdd),然后再解析为Date。Java8中可以用更多的方法来实现这个需求,下面使用三种方法:使用Format方法、 使用Of方法和使用With方法,性能对比,使用
Python进阶者 Python进阶者
11个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这
Easter79
Easter79
Lv1
今生可爱与温柔,每一样都不能少。
文章
2.8k
粉丝
5
获赞
1.2k