當(dāng)前位置:首頁 > 公眾號(hào)精選 > 小林coding
[導(dǎo)讀]—?1?— 需求 有時(shí)候希望賦值運(yùn)算符兩邊的類型可以不匹配。 比如:把一個(gè) ?int? 類型變量賦值給一個(gè) Complex(復(fù)數(shù))對(duì)象,或把一個(gè) ?char*? 類型的字符串賦值給一個(gè)字符串對(duì)象,此時(shí)就需要重載?=?賦值運(yùn)算符?。 需要注意的是:賦值運(yùn)算符?=?只能重載為成員函


 1 

需求


有時(shí)候希望賦值運(yùn)算符兩邊的類型可以不匹配。


比如:把一個(gè)  int  類型變量賦值給一個(gè) Complex(復(fù)數(shù))對(duì)象,或把一個(gè)  char*  類型的字符串賦值給一個(gè)字符串對(duì)象,此時(shí)就需要重載 = 賦值運(yùn)算符 


需要注意的是:賦值運(yùn)算符 = 只能重載為成員函數(shù)。




 2 

舉個(gè)栗子


下面我們以自定義一個(gè)自己的字符串類代碼的例子,講解賦值運(yùn)算符的重載函數(shù)。


MyString 字符串類所需的成員函數(shù):

  • 構(gòu)造函數(shù) / 析構(gòu)函數(shù)
  • 返回 char* 指針的函數(shù)
  • 賦值運(yùn)算符重載函數(shù)


出結(jié)果:


Hello~Hi~


重載 =號(hào)運(yùn)算符函數(shù)后,s = "Hello~"; 語句就等價(jià)于 s.operator=("Hello~");


需要注意的一點(diǎn)是:


上面的 MyString s2 = "Hello!";  語句實(shí)際上是初始化語句,而不是賦值語句,因?yàn)槭浅跏蓟Z句,所以需要調(diào)用構(gòu)造函數(shù)進(jìn)行初始化,那么這時(shí)就需要有 char* 參數(shù)的構(gòu)造函數(shù),由于我們沒有定義此構(gòu)造函數(shù),所以就會(huì)編譯出錯(cuò)。





 3 

淺拷貝和深拷貝


還是依據(jù)上面的例子,假設(shè)我們要實(shí)現(xiàn)最后一個(gè)語句的方式:


MyString s1,s2;s1 = "this"; // 調(diào)用重載的賦值語句s2 = "that"; // 調(diào)用重載的賦值語句s1 = s2; // 如何實(shí)現(xiàn)這個(gè)??


s1 = s2; 語句目的希望是 s1 對(duì)象放的字符串和 s2 對(duì)象放的字符串要一樣,由于 = 號(hào)兩邊的類似都是對(duì)象,編譯器會(huì)用原生的賦值運(yùn)算符函數(shù)。


但是這個(gè)原生的賦值運(yùn)算符函數(shù)對(duì)于有指針成員變量的對(duì)象來說,是非常危險(xiǎn)的!



— —


淺拷貝


如果用原生的賦值運(yùn)算符函數(shù)去賦值有指針成員變量的對(duì)象,就會(huì)使得兩個(gè)對(duì)象的指針地址也是一樣的,也就是兩個(gè)對(duì)象的指針成員變量指向的地址是同一個(gè)地方,這種方式就是淺拷貝。


這時(shí)當(dāng)一個(gè)對(duì)象釋放了指針成員變量時(shí),那么另外一個(gè)對(duì)象的指針成員變量指向的地址就是空的了,再次使用這個(gè)對(duì)象時(shí),程序就會(huì)奔潰了,因?yàn)樵搶?duì)象的指針成員函數(shù)已經(jīng)是個(gè)不合法的指針了!




— —


深拷貝


如果對(duì)象里面有指針成員變量,則我們需要對(duì)原生的賦值運(yùn)算符函數(shù),防止出現(xiàn)程序出錯(cuò)現(xiàn)象的發(fā)生。


因此要在 class MyString 類里加上如下成員函數(shù):


MyString & operator=(const MyString & s){ // 釋放舊字符串資源 delete [] m_str;
// 生成新字符串的空間大小,長度多+1的目的是存放\0 m_str = new char[strlen(s.m_str) +1 ];
// 拷貝新字符串的內(nèi)容 strcpy(m_str, s.m_str);
// 返回該對(duì)象 return *this;}



這么做就夠了嗎?還有什么需要改進(jìn)的地方嗎?我們?cè)诳紤]下面的語句:


MyString s;s = "Hello";s = s; // 是否會(huì)有問題?

最后一個(gè)語句是否會(huì)有問題?


s = s;等價(jià)于s.operator=(s),由于s和s是相同的對(duì)象,那么就沒必要完全執(zhí)行重載的賦值 = 的函數(shù)了。


我們?cè)偌觽€(gè)判斷,當(dāng)左右兩邊是相同對(duì)象時(shí),就直接返回該對(duì)象就好:


MyString & operator=(const MyString & s){ // 當(dāng)左右兩邊是相同對(duì)象時(shí),就直接返回該對(duì)象就 if(this == &s) return *this;
delete [] m_str; m_str = new char[strlen(s.m_str) +1 ]; strcpy(m_str, s.m_str); return *this;}





 4 

返回值討論


對(duì) opterator= 返回值討論:

  • void 好不好?
  • MyString 好不好?
  • 為什么是MyString &


當(dāng)我們重載一個(gè)運(yùn)算符的時(shí)候,好的風(fēng)格應(yīng)該是盡量保留運(yùn)算符原本的特性。


考慮:

  • a = b = c; 這個(gè)賦值語句的順序是先 b = c ,然后在 a = (b = c) 。如果返回的 void 類型,那么 a = (void) 顯然是不成立的;
  • (a = b) = c; 這個(gè)賦值語句會(huì)修改 a 的值,如果返回的類型是 MyString 對(duì)象,那么就無法修改 a 的值了。


分別等價(jià)于:

  • a.operator=(b.operator=(c));
  • (a.operator=(b)).operator=(c);


所以綜上考慮, operator= 返回值類型是 MyString & 是比較好的。





 5 

復(fù)制構(gòu)造函數(shù)


上面的 MyString 類是否就沒有問題了?


MyString s;s = "Hello";MyString s1(s); // 要考慮這種情況,那就要重載復(fù)制(拷貝)構(gòu)造函數(shù)


如果使用默認(rèn)的復(fù)制(拷貝)構(gòu)造函數(shù),那就對(duì)有指針成員變量的對(duì)象會(huì)有問題,因?yàn)闀?huì)默認(rèn)的復(fù)制(拷貝)構(gòu)造函數(shù)會(huì)導(dǎo)致兩個(gè)對(duì)象的指針成員變量指向同一個(gè)的空間。


所以需要對(duì)復(fù)制(拷貝)構(gòu)造函數(shù)重載,并實(shí)現(xiàn)深拷貝的方式:


MyString (const MyString &s){ m_str = new char[strlen(s.m_str) + 1]; strcpy(m_str, s.m_str);}





 6 

小結(jié)

所以最后的 MyString 字符串類所需的成員函數(shù):

  • 構(gòu)造函數(shù) / 析構(gòu)函數(shù)
  • 返回 char* 指針的函數(shù)
  • 賦值運(yùn)算符重載函數(shù)(深拷貝
  • 復(fù)制構(gòu)造函數(shù)(深拷貝





推薦閱讀:
C++ 運(yùn)算符重載的基本概念


  小林coding  



免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場(chǎng),如有問題,請(qǐng)聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉