深入解析Swift編程中的構(gòu)造方法
一、引言
構(gòu)造方法是一個(gè)類創(chuàng)建對(duì)象最先也是必須調(diào)用的方法,在Objective-C中,開發(fā)者更習(xí)慣稱這類方法為初始化方法。在Objective-C中的初始化方法與普通函數(shù)相比除了要以init抬頭外并無(wú)太嚴(yán)格的分界,而在Swift語(yǔ)言體系中,構(gòu)造方法與普通的方法分界十分嚴(yán)格,從格式寫法上就有不同,普通方法函數(shù)要以func聲明,構(gòu)造方法統(tǒng)一為init命名,不需要func關(guān)鍵字聲明,不同的構(gòu)造方法采用方法重載的方式創(chuàng)建。
二、構(gòu)造方法的復(fù)寫與重載
在Objective-C中,不同的初始化方法就是不同的函數(shù),這便不存在方法重載的概念。Swift中要?jiǎng)?chuàng)建自定義的構(gòu)造方法,需要開發(fā)者對(duì)init構(gòu)造方法進(jìn)行重載操作。任何一個(gè)自定義的類,只要其有父類,除了可以繼承下來(lái)父類已有的構(gòu)造方法外,還可以復(fù)寫父類的構(gòu)造方法,使其適用于自身。和Objective-C類似,復(fù)寫父類的構(gòu)造方法時(shí),要在其中調(diào)用父類的構(gòu)造方法,重載可以理解為一種特殊的復(fù)寫父類構(gòu)造方法,因此在重載的構(gòu)造方法中也要調(diào)用父類的構(gòu)造方法。
創(chuàng)建一個(gè)繼承于NSObject的類,復(fù)寫構(gòu)造方法,代碼示例如下:
class ClassOne: NSObject { //聲明一個(gè)本類特有的常量 var tip:Int //復(fù)寫父類的構(gòu)造方法 需要用override關(guān)鍵字 override init() { //構(gòu)造方法中要對(duì)所有成員常量完成創(chuàng)建 tip = 1; //在創(chuàng)建完所有成員常量后 調(diào)用父類構(gòu)造方法 super.init() } //重載構(gòu)造方法1 init(one:Int){ tip=one super.init() } //重載構(gòu)造方法2 使用convenience關(guān)鍵字進(jìn)行修飾 convenience init(two:String) { //使用convenience關(guān)鍵字進(jìn)行修飾的構(gòu)造方法要調(diào)用本類的構(gòu)造方法進(jìn)行 self.init(one: two.characters.count) } //重載構(gòu)造方法3 使用required關(guān)鍵字進(jìn)行修飾 使用required關(guān)鍵字進(jìn)行修飾的構(gòu)造方法子類必須繼承或復(fù)寫 required init(three:Float) { tip=10 super.init() }
上面示例代碼中,不帶參數(shù)的init()方法為復(fù)寫父類的方法,因此需要使用關(guān)鍵字override來(lái)修飾。重載構(gòu)造方法1帶一個(gè)Int類型的 參數(shù),父類中并沒(méi)有這個(gè)構(gòu)造方法,但是在其實(shí)現(xiàn)中,依然需要調(diào)用父類中的某個(gè)構(gòu)造方法完成。構(gòu)造方法2是一個(gè)帶String類型參數(shù)的構(gòu)造方法,其用convenience關(guān)鍵字為構(gòu)造方法的一個(gè)修飾關(guān)鍵字,后面會(huì)介紹。構(gòu)造方法3為一個(gè)帶Float類型參數(shù)的構(gòu)造方法,但其使用required關(guān)鍵字進(jìn)行了修飾,使用required關(guān)鍵字進(jìn)行修飾的構(gòu)造方法子類必須繼承或者復(fù)寫。構(gòu)造方法1,2,3都是對(duì)init()構(gòu)造方法的一種重載,但卻是3中類型全然不同的構(gòu)造方法。
三、Designated構(gòu)造方法與Convenience構(gòu)造方法
Swift中的構(gòu)造方法分為Designated構(gòu)造方法與Convenience構(gòu)造方法兩類,Designated構(gòu)造方法也被稱為指定構(gòu)造方法,Convenience構(gòu)造方法也被稱為方便構(gòu)造方法。Designated構(gòu)造方法不加任何修飾關(guān)鍵字,Convenience構(gòu)造方法需要使用Convenience關(guān)鍵字進(jìn)行修飾??梢赃@樣理解,Convenience類型的構(gòu)造方法是為了方便使用從Designated構(gòu)造方法中分支出來(lái)的構(gòu)造方法,官方文檔中有如下描述:
1.子類Designated構(gòu)造方法中必須調(diào)用父類的Designated構(gòu)造方法。
2.Convenience構(gòu)造方法中必須調(diào)用當(dāng)前類的構(gòu)造方法。
3.Convenience構(gòu)造方法歸根結(jié)底要調(diào)用到Designated構(gòu)造方法。
官方文檔的一張圖可以清晰的描述上述關(guān)系:
四、構(gòu)造方法的繼承關(guān)系
關(guān)于子類繼承父類的構(gòu)造方法有這樣幾個(gè)特性:
1.如果子類沒(méi)有復(fù)寫任何父類的構(gòu)造方法,則默認(rèn)子類將繼承所有父類的構(gòu)造方法,包括Designated構(gòu)造方法與Convenience構(gòu)造方法。
2.如果子類復(fù)寫了父類某一構(gòu)造方法,則子類默認(rèn)不在繼承所有父類的構(gòu)造方法,對(duì)于Designated類型的構(gòu)造方法,子類復(fù)寫了哪些,哪些才能夠被使用,對(duì)于Convenienve類型的構(gòu)造方法,子類復(fù)寫的其調(diào)用的Designated構(gòu)造方法后會(huì)被自動(dòng)繼承。
3.如果父類中的構(gòu)造方法是required修飾的,則子類必須進(jìn)行繼承或復(fù)寫。
曾經(jīng)有朋友和我抱怨,Objective-C中的繼承是一種十分不人性,它強(qiáng)制子類繼承所有父類的方法與屬性無(wú)論子類是否需要,分析上面的一些規(guī)則可以發(fā)現(xiàn),Swift與Objective-C相比,在構(gòu)造方法方面語(yǔ)法會(huì)更加嚴(yán)格,這樣做在編程上更加安全。在Objective-C中,子類將被強(qiáng)制繼承所有父類的初始化方法,這樣開發(fā)者在使用時(shí)常常會(huì)出現(xiàn)疑惑,有時(shí)一個(gè)子類往往有特定的初始化方法,僅僅通過(guò)父類的初始化方法不能夠正確的完成初始化,在編程時(shí),往往需要特殊注釋來(lái)提示開發(fā)者。Swift設(shè)定的這些構(gòu)造方法原則可以將無(wú)關(guān)的父類構(gòu)造方法剔除在外,在編程時(shí)更加嚴(yán)格安全,減少疑惑與不可控因素。
五、構(gòu)造方法的實(shí)現(xiàn)原則
無(wú)論Designated類型的構(gòu)造方法還是Convenience類型的構(gòu)造方法,只要其有父類,最終都要實(shí)現(xiàn)父類的Designated構(gòu)造方法。Swift語(yǔ)言要求,在構(gòu)造方法中要完成所有成員常量或者變量的構(gòu)造或賦值(optional值除外)。在對(duì)成員常量或變量進(jìn)行構(gòu)造賦值時(shí),要在調(diào)用父類的初始化方法之前,這里還有一點(diǎn)需要注意,父類的成員屬性也會(huì)被子類繼承,如果要在子類復(fù)寫的父類方法中對(duì)繼承來(lái)的父類成員屬性進(jìn)行重新構(gòu)造或賦值,則必須在調(diào)用父類構(gòu)造方法之后,例如創(chuàng)建ClassTwo類繼承于ClassOne,復(fù)寫方法如下:
class ClassTwo: ClassOne { //子類自己的屬性 let tipTwo:Int override init() { //調(diào)用父類構(gòu)造方法前進(jìn)行自己屬性的構(gòu)造 tipTwo = 1 //調(diào)用父類構(gòu)造方法 super.init() //對(duì)從父類繼承來(lái)的屬性進(jìn)行重構(gòu)造 tip = 1000; } required init(three: Float) { fatalError("init(three:) has not been implemented") } }
Swift語(yǔ)言這種強(qiáng)制化得構(gòu)造規(guī)則,能夠保證一個(gè)類在完成構(gòu)造時(shí),其內(nèi)部的所有屬性都構(gòu)造完成。在使用Objective-C進(jìn)行開發(fā)時(shí),很多初學(xué)者都可能會(huì)遇到這樣一種情況,完成了某個(gè)類的初始化,但向類的屬性進(jìn)行賦值時(shí)卻沒(méi)有成功,因?yàn)镺bjective-C中并沒(méi)有這樣的語(yǔ)法,在類初始化成功后,其屬性是否初始化了完全取決于開發(fā)者,Swift優(yōu)化了這一設(shè)計(jì)。
綜上可以了解,Swift語(yǔ)言雖然更加嚴(yán)格,卻將更多本來(lái)需要開發(fā)者注意的地方交由了編譯器,實(shí)際上是減輕了開發(fā)者的負(fù)擔(dān)。
相關(guān)文章
Swift中的條件判斷、循環(huán)、跳轉(zhuǎn)語(yǔ)句基礎(chǔ)學(xué)習(xí)筆記
if、for和while循環(huán)、switch等這些基本的程序流程控制語(yǔ)句基本上是每個(gè)編程語(yǔ)言的標(biāo)配,在入門環(huán)節(jié)中,這里對(duì)Swift中的條件判斷、循環(huán)、跳轉(zhuǎn)語(yǔ)句基礎(chǔ)學(xué)習(xí)筆記作了一個(gè)整理:2016-06-06Swift中動(dòng)態(tài)調(diào)用實(shí)例方法介紹
這篇文章主要介紹了Swift中動(dòng)態(tài)調(diào)用實(shí)例方法介紹,在Swift中有一類很有意思的寫法,可以讓我們不直接使用實(shí)例來(lái)調(diào)用這個(gè)實(shí)例上的方法,而是通過(guò)類型取出這個(gè)類型的某個(gè)實(shí)例方法的簽名,然后再通過(guò)傳遞實(shí)例來(lái)拿到實(shí)際需要調(diào)用的方法,需要的朋友可以參考下2015-01-01Swift使用transform 實(shí)現(xiàn)重復(fù)平移動(dòng)畫效果
這篇文章主要介紹了Swift使用transform 實(shí)現(xiàn)重復(fù)平移動(dòng)畫效果,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-07-07在Swift中使用Objective-C編寫類、繼承Objective-C類
這篇文章主要介紹了在Swift中使用Objective-C編寫類、繼承Objective-C類等操作方法介紹,需要的朋友可以參考下2014-07-07switch多選擇結(jié)構(gòu)、循環(huán)結(jié)構(gòu)示例詳解
這篇文章主要介紹了switch多選擇結(jié)構(gòu)、循環(huán)結(jié)構(gòu),本文結(jié)合示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-12-12Swift 基本數(shù)據(jù)類型詳解總結(jié)
在我們使用任何程序語(yǔ)言編程時(shí),需要使用各種數(shù)據(jù)類型來(lái)存儲(chǔ)不同的信息。變量的數(shù)據(jù)類型決定了如何將代表這些值的位存儲(chǔ)到計(jì)算機(jī)的內(nèi)存中。在聲明變量時(shí)也可指定它的數(shù)據(jù)類型。所有變量都具有數(shù)據(jù)類型,以決定能夠存儲(chǔ)哪種數(shù)據(jù)2021-11-11