類的作者經(jīng)常要定義一些輔助函數(shù),盡管這些函數(shù)定義的操作從概念上來(lái)說(shuō)屬于類的接口的組成部分凉逛,但它們實(shí)際上并不屬于類本身。?
定義非成員函數(shù)妆够,通常把函數(shù)的聲明和定義分離開神妹。如果函數(shù)在概念上屬于類但是不定義在類中家妆,則它一般應(yīng)與類聲明在同一個(gè)頭文件內(nèi)。
這種方式下使用接口的任何部分都只需引入一個(gè)文件蛹找。
read 函數(shù)從給定流中將數(shù)據(jù)讀到給定的對(duì)象里。
print 函數(shù)則負(fù)責(zé)將給定對(duì)象的內(nèi)容打印到給定的流中届慈。
read、print 分別介紹一個(gè)各自 IO 類型的引用作為其參數(shù)忿偷,因?yàn)?IO 類屬于不能被拷貝的類型金顿,所以只能通過(guò)引用來(lái)傳遞它們。且因?yàn)樽x寫的操作都會(huì)改變流的內(nèi)容鲤桥,所以兩個(gè)函數(shù)接收的都是普通引用揍拆。
print 函數(shù)不負(fù)責(zé)換行,執(zhí)行輸出任務(wù)的函數(shù)應(yīng)該盡量減少對(duì)格式的控制芜壁,這樣可以確保由用戶代碼來(lái)決定是否換行礁凡。
每個(gè)類都分別定義了它的對(duì)象被初始化的方式,類通過(guò)一個(gè)或幾個(gè)特殊的成員函數(shù)來(lái)控制其對(duì)象的初始化過(guò)程慧妄,這些函數(shù)叫做構(gòu)造函數(shù)顷牌。
構(gòu)造函數(shù)的任務(wù)是初始化類對(duì)象的數(shù)據(jù)成員,無(wú)論何時(shí)只要類的對(duì)象被創(chuàng)建塞淹,就會(huì)執(zhí)行構(gòu)造函數(shù)窟蓝。
構(gòu)造函數(shù)的名字和類名相同。構(gòu)造函數(shù)沒(méi)有返回類型饱普。
構(gòu)造函數(shù)有一個(gè)可以為空的參數(shù)列表和一個(gè)可以為空的函數(shù)體运挫。
類可以包含多個(gè)構(gòu)造函數(shù)状共,和其它重載函數(shù)差不多,不同的構(gòu)造函數(shù)之間必須在參數(shù)數(shù)量或參數(shù)類型上有所區(qū)別谁帕。
構(gòu)造函數(shù)不能被聲明成 const 峡继,當(dāng)創(chuàng)建類的一個(gè) const 對(duì)象時(shí),指導(dǎo)構(gòu)造函數(shù)完成初始化過(guò)程匈挖,對(duì)象才能真正取得常量屬性儡循。構(gòu)造函數(shù)在 const 對(duì)象的構(gòu)造過(guò)程中可以向其寫值择膝。
類通過(guò)一個(gè)特殊的構(gòu)造函數(shù)來(lái)控制默認(rèn)初始化的過(guò)程腹侣,這個(gè)函數(shù)叫做默認(rèn)構(gòu)造函數(shù)筐带,默認(rèn)構(gòu)造函數(shù)不需要實(shí)參缤灵。
如果類沒(méi)有顯式的定義構(gòu)造函數(shù),編譯器則會(huì)隱式的定義一個(gè)構(gòu)造函數(shù)芝薇。
編譯器創(chuàng)建的構(gòu)造函數(shù)也被稱為合成的默認(rèn)構(gòu)造函數(shù)馋劈。如果存在類內(nèi)初始值晾嘶,用它來(lái)初始化成員械姻,否則默認(rèn)初始化該成員楷拳。
對(duì)于普通或較為復(fù)雜的類,必須定義一個(gè)它自己的默認(rèn)構(gòu)造函數(shù)陶耍。
對(duì)于 sales_data 類物臂,使用以下四個(gè)參數(shù)定義不同的構(gòu)造函數(shù)棵磷。
一個(gè) istream& 從中讀取一條交易信息仪媒。
一個(gè) const string& 表示 ISBN 編號(hào)、一個(gè) unsigned 表示售出的圖書數(shù)量偎巢、一個(gè) double 表示圖書的售出價(jià)格压昼。
一個(gè) const string& 表示 ISBN 編號(hào)窍霞,編譯器將賦予其它成員默認(rèn)值但金。
一個(gè)空參數(shù)列表(默認(rèn)構(gòu)造函數(shù)),因?yàn)橐呀?jīng)定義了其它構(gòu)造函數(shù)似枕,所以必須定義一個(gè)默認(rèn)構(gòu)造函數(shù)菠净。
解釋默認(rèn)構(gòu)造函數(shù)的含義:sales_data() = defaule;
因?yàn)闃?gòu)造函數(shù)不接受任何實(shí)參牵咙,所以它是一個(gè)默認(rèn)構(gòu)造函數(shù)。定義這個(gè)構(gòu)造函數(shù)僅僅是因?yàn)榧刃枰渌问降臉?gòu)造函數(shù)也需要默認(rèn)的構(gòu)造函數(shù)另凌。
如果需要默認(rèn)的行為吠谢,那么可以通過(guò)在參數(shù)列表后面寫上 = default 來(lái)要求編譯器生成構(gòu)造函數(shù)工坊。= default 既可以和聲明一起出現(xiàn)在類內(nèi)部王污,也可以作為定義出現(xiàn)在類的外部昭齐。如過(guò)在內(nèi)部構(gòu)造函數(shù)時(shí)內(nèi)聯(lián)的,在類的外部則該成員默認(rèn)情況下不是內(nèi)聯(lián)的把沼。
冒號(hào)及冒號(hào)和花括號(hào)之間的代碼,花括號(hào)定義了空的函數(shù)體捆愁。
其中新出現(xiàn)的部分稱為構(gòu)造函數(shù)初始值列表昼丑,其負(fù)責(zé)為新創(chuàng)建的對(duì)象的一個(gè)或幾個(gè)數(shù)據(jù)成員賦初值菩帝。
構(gòu)造函數(shù)初始值是成員名字的一個(gè)列表呼奢,每個(gè)名字后面緊跟括號(hào)或在花括號(hào)內(nèi)括起來(lái)的成員初始值辐董。
不同成員的初始化通過(guò)逗號(hào)分隔開來(lái)简烘。
以 istream 為參數(shù)的構(gòu)造函數(shù)需要執(zhí)行一些實(shí)際的操作孤澎。
構(gòu)造函數(shù)沒(méi)有返回類型。
當(dāng)在類的外部定義構(gòu)造函數(shù)時(shí)贱迟,必須指明該構(gòu)造函數(shù)是哪個(gè)類的成員茶敏。因此 sales_data::sales_data 的含義是我們定義 sales_data 類的成員惊搏,它的名字是 sales_data恬惯,又因?yàn)槌蓡T名字和類名相同酪耳,所以是一個(gè)構(gòu)造函數(shù)。
這個(gè)構(gòu)造函數(shù)的構(gòu)造函數(shù)初始值列表是空的梢夯,但是由于執(zhí)行了構(gòu)造函數(shù)體颂砸,所以對(duì)象的成員仍能被初始化。
沒(méi)有出現(xiàn)在構(gòu)造函數(shù)初始值列表中的成員將通過(guò)相應(yīng)的類內(nèi)初始值初始化知允,或者執(zhí)行默認(rèn)初始化温鸽。
sales_data 函數(shù)一旦開始執(zhí)行涤垫,bookNo 將初始化成 string 對(duì)象蝠猬, units_sold 、revenue 將是 0 匆绣。
除了定義類的對(duì)象如何初始化之外崎淳,類還需要控制拷貝拣凹、賦值嚣镜、銷毀對(duì)象時(shí)發(fā)生的行為菊匿。
對(duì)象在幾種情況下會(huì)被拷貝:初始化變量以及以值得方式傳遞或返回一個(gè)對(duì)象等....當(dāng)使用賦值運(yùn)算符時(shí)會(huì)發(fā)生對(duì)象得賦值操作捧请。當(dāng)對(duì)象不再存在時(shí)執(zhí)行銷毀操作。
若不主動(dòng)定義這些操作力麸,編譯器將替我們合成它們闺鲸。編譯器生成得版本將對(duì)對(duì)象得每個(gè)成員執(zhí)行拷貝摸恍、賦值立镶、銷毀媚媒。