【C++類和對象解密】面向對象編程的核心概念(下)

????????之前我們了解到構造函數是在對象實例化之時對對象完成初始化工作的一個函數。在我們不寫時,編譯器會自動生成構造函數。構造函數有一些特點,比如,他對內置類型不做處理,對自定義類型的成員會去調用其自身的構造。

????????我們上篇文章還提到了,默認成員函數不僅僅指我們不寫編譯器自動生成的函數,當我們不傳參數,編譯器會自動調用的函數,均為默認成員函數。

一、再探構造函數

  • 之前我們實現構造函數時,初始化成員變量主要使用函數體內賦值,構造函數初始化還有一種方式,就是初始化列表,初始化列表的使用方式是以一個冒號開始,接著是一個逗號分隔的數據成員列表,每個“成員變量”后邊跟一個放在括號中的初始值或表達式。
  • 語法上理解,初始化列表可以認為是每個成員變量定義初始化的地方。

??????※※每個成員變量在初始化列表中只能出現一次,這個很重要,編譯會報錯。

  • 當然,函數體內的初始化和初始化列表初始化是可以同時存在的,這里會有同學疑惑,既然函數體內也可以進行初始化,為什么要延伸出初始化列表這個概念呢?要記住,存在即合理
  • 引用成員變量const成員變量沒有默認構造的類類型變量,必須放在初始化列表位置進行初始化,否則編譯會報錯。
  • 首先,我們來看,引用成員變量const成員變量有啥相似,那就是這兩者必須在定義時就初始化

如下圖,當在函數體內對這兩種變量進行初始化時,編譯會報錯。

然而,當我們試圖修改這種初始化方式,試圖將_ref變為year的別名,_i變為day的別名,雖然從下圖可以看出,語法上是沒什么問題的,可以運行成功,但是,要考慮到year、month、day均是形參,當出了函數作用域,他們均將被銷毀,這時的_ref、_i就變成了野引用,相當于野指針,是極其危險的。

  • 同時,第三個特殊的變量是沒有默認構造的類類型變量需要在定義時顯式地調用構造函數傳參給值所以也需要在定義之時進行初始化。這里要注意一點,如果圖中的Time類的構造函數有缺省值,就不需要在初始化列表進行初始化了。

  • 嚴格來說,這些聲明過的成員變量不管是否在初始化列表中出現都會走一遍初始化列表。
  • 還有一點,我們可以看到_day編譯器只賦了一個隨機值,這也是我們之前說過的C++對內置類型的初始化是不做處理的,事實上,_day也走了初始化列表。

引用成員變量的初始化可以參考下圖:

在聲明處也可以這樣寫,相當于給成員變量缺省值


※※※按聲明順序(與初始化列表出現順序無關),所以建議同學們將聲明順序與初始化列表順序保持一致。

這里要問大家一個問題:這里最終的初始化結果是什么?

答案是_i = 1。

  • 首先,我們要明確,_i如果在聲明處給了缺省值3,假設我們沒有在初始化列表中顯式對_i進行初始化,_i還是會走初始化列表,但程序是沒有錯誤的。
  • 其次,由于_i在初始化列表進行了初始化,且值為1,我們可以從上面給大家提供的邏輯分析,由于成員變量已經顯式初始化,就不會再考慮未初始化的情形了。

???????

??有同學又要問,我們可不可以只用初始化列表初始化,不在函數體內部了,當然是不行的了!還是那句話:存在即合理!像這一類,需要函數邏輯來進行初始化的成員,當然還是需要在函數體內進行初始化。

建議之后初始化將缺省值、初始化列表、函數體結合起來共同實現,因為我們的成員變量(包括未顯式初始化的成員)都要走一遍初始化列表,有些初始化邏輯又必須使用函數,我們何必要浪費資源,不如都應用起來!~

?二、類型轉換

????????C語言階段我們也曾提到過類型轉換,內置類型隱式轉換,前提是他們之間是有關聯的,比如整型之間的轉換,int可以轉換為short;比如整形和浮點數之間也可以進行相互轉化,int轉換為double;再比如整型和指針的轉換;指針和指針之間也可以相互轉換。

  • C++支持內置類型隱式轉換為類類型對象,需要有相關內置類型為參數的構造函數

因此,也就引申出了:

有同學會問,這種場景現實嗎?由于r1是別名,這里的1隱式轉換為A,為臨時對象,其實是可以的。但臨時對象具有常性,這是我們之前就了解過的,所以這里應該在A前加const。

?在之前的學習中,我們了解到,類型轉換會構造臨時對象,臨時對象又具有常性。

  • 構造函數前面加explicit就不再支持隱式類型轉換。

同時,要注意一個問題,當我們對多內置類型的對象傳值時,使用(1,1)會被誤認為是逗號表達式,從而只傳了一個值,在此,我們可以使用{1,1}。

  • 類類型的對象之間也可以隱式類型轉換,需要相應的構造函數支持。

三、static成員

??????提出一個需求,實現一個類,計算程序中創建出了多少個類對象?最能想到的方式就是定義一個全局變量_scount,每當創建一個對象調用一次拷貝構造,_scount就加1。但我們上面提到過,當編譯器同時遇到連續構造和拷貝構造,就會采用優化,變為直接構造,除非關閉優化。所以到底程序創建了多少對象,是算不明白的。

還有一個弊端,_scount作為全局變量,在任何類、函數里都可以修改,是否可以定義一個專屬于一個類的全局變量呢?

答案是可以的。這個變量就叫做靜態成員變量,它不屬于某個對象,而是屬于整個類,屬于這個類的所有對象,相當于“類中的全局變量”。屬于靜態區,不存在對象中,并且受訪問限定符的限制,不會輕易改變。

再一個問題,static靜態成員變量是不可以給缺省值的,由于它不參與對象的創建,不會走初始化列表,更不會使用缺省值。

※※※靜態成員變量要在類內聲明,類外定義。

類中的靜態成員變量怎樣訪問呢?

假設設置他為公有,那么只要指定類域or用對象訪問就可以使用(因為這兩種情況都可以讓編譯器識別到_scount這個靜態變量是屬于A類的),但是會遇到與上面提到過的全局變量一樣的問題,有隨時被修改的風險;

假設設置為私有,我們可以設置一個公有的成員函數,有點類似于Java中的get/set()。

?除了靜態成員變量,還有靜態成員函數。所謂靜態,就是在函數返回類型前加上static。

※很重要的一點,靜態成員函數的特點是沒有this指針!

也就是如果對象中有非靜態的成員變量,在靜態成員函數中是不能訪問的!

靜態成員函數應用場景:

解鎖訪問靜態成員函數的兩種姿勢:

	cout << A::Get() << endl;cout << aa1.Get() << endl;

四、友元?

  • 我們在類外面是不能訪問私有或保護成員的,友元則提供了一種突破類訪問限定符封裝的方式。
  • 友元分為:友元函數友元類
  • 在函數聲明或者類聲明的前面加friend,并且把友元聲明放到一個類的里面。
  • 外部友元函數可訪問類的私有和保護成員,友元函數僅僅是一種聲明,他不是類的成員函數。
  • 友元函數可以在類定義的任何地方聲明,不受訪問限定符限制。
  • 一個函數可以是多個類的友元函數。

  • 友元類的成員函數都可以是另一個類的友元函數,都可以訪問另一個類中的私有和保護成員。
  • 友元類的關系是單向的,不具有交換性,比如A類是B類的友元,但B類不是A類的友元。

  • 友元類不能傳遞,如果A是B的友元,B是C的友元,但A不是C的友元。
  • 友元有時提供了便利,但是友元會增加耦合度,破壞了C++的封裝特性,所以友元不宜多用。?

五、內部類

  • 如果一個類定義在另一個類內部,這個內部類就叫做內部類。內部類是一個獨立的類,跟定義在全局相比,他只是受外部類的類域限制訪問限定符限制,所以外部類定義的對象中不包含內部類。
  • 如果內部類在外部類中定義為私有,那么這個內部類就是外部類的專屬類。
  • 內部類默認是外部類的友元。
  • 這里說明,內部類在外部類的類域里面,突破類域,可以直接訪問靜態成員變量
  • 由于B(內部類)默認是A(外部類)的友元,所以非靜態成員變量需要通過調用對象進行訪問
  • 內部類本質也是一種封裝,當A類跟B類緊密關聯,A類實現出來主要就是給B類使用,那么可以考慮把A類設計為B的內部類,如果放到private/protected位置,那么A類就是B類的專屬內部類,其他地方都用不了。

六、匿名對象

  • 匿名對象,從他的名字可以看出,他是沒有名字的對象,也就是在定義時不起名字;我們在之前定義有名對象時,曾提到調用無參構造時不要寫括號,會與函數聲明沖突,分不清楚,與有名對象不同,匿名對象在調用無參構造時要加上括號,如果不加,就很容易迷惑人,不知道在寫什么。
  • 它是由我們主動寫的,并非編譯器生成的。
  • 匿名對象還有一個顯著的特點:它的生命周期只在當前一行。即用完就被銷毀掉了。

七、對象拷貝時的編譯器優化

  • 現代編譯器會為了盡可能提高程序的效率,在不影響正確性的情況下,盡可能減少一些傳參和傳返回值的過程中可以省略的拷貝。
  • 對于如何優化,C++標準并沒有嚴格規定,各個編譯器會根據情況自行處理。當前主流、相對新一點的編譯器對于連續一個表達式步驟中的連續拷貝會進行合并優化,有些更新的編譯器還會進行跨行跨表達式的合并優化。

(傳值傳參)以下兩種情況,將兩個構造過程合二為一,提高了程序效率:


(傳值返回)

總結一下:

  • 如果用對象向函數進行傳值傳參,盡可能用匿名對象or隱式類型轉換的方式來替代有名對象(無優化)
  • 如果傳值返回,接收返回值更推薦使用拷貝構造的方式(也就是上上圖的第一種方式)

????????未完待續…點個贊唄~~ (2025/7/17/20:03:19)

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/89355.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/89355.shtml
英文地址,請注明出處:http://en.pswp.cn/web/89355.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Flutter基礎(前端教程①②-序列幀動畫)

&#x1f9e0; 核心思路總結??徹底繞過 Image組件重建帶來的性能瓶頸??&#xff1a;不再讓 setState重建包含 Image的 Widget 樹&#xff08;這是開銷大、可能導致閃爍的根源&#xff09;&#xff0c;改為使用底層畫布 (Canvas) 直接繪制預先處理好的圖像幀數據。好的&…

Qt添加dmp文件生成及pdb文件

1.Pdb文件生成 下圖先通過構建生成Pdb文件&#xff0c;然后運行程序&#xff0c;通過提前準備的崩潰按鈕使得程序崩潰&#xff0c;生成“dump文件”的演示。 # #添加dmp文件生成及pdb文件生成DEFINES QT_MESSAGELOGCONTEXT DEFINES QT_DEPRECATED_WARNINGS# # 添加DUMP文件…

opencv、torch、torchvision、tensorflow的區別

一、框架定位與核心差異PyTorch動態計算圖&#xff1a;實時構建計算圖支持Python原生控制流&#xff08;如循環/條件&#xff09;&#xff0c;調試便捷。學術主導&#xff1a;2025年工業部署份額24%&#xff0c;適合快速原型開發&#xff08;如無人機自動駕駛、情緒識別&#x…

離散與組合數學 雜記

生成函數 概念 又稱母函數把一個無窮數列 {an}\{a_n\}{an?}&#xff08;默認從 000 項起&#xff09;表示成 G(x)∑i≥0aixiG(x)\displaystyle\sum_{i\ge0} a_ix^iG(x)i≥0∑?ai?xi 的函數形式。例如&#xff1a; ai2ia_i2^iai?2i&#xff1a;G(x)∑i≥02ixiG(x)\display…

學習OpenCV---顯示圖片

學習OpenCV—顯示圖片 最近在學習OpenCV入門&#xff0c;于是記錄一下自己的學習過程。 一、配置環境 第一步 從官方網站中下載OpenCV開源庫。官方下載網站 打開官網后&#xff0c;能看到有很多的版本。我個人下載的是4.11.0版本。點擊圖中的下載 下載完成后&#xff0c;解…

第一次接觸自動化監測,需要付費廠家安裝服務嗎?比人工測量主要區別是啥?

人工檢測是依靠目測檢查或借助于便攜式儀器測量得到的信息&#xff0c;但是隨著整個行業的發展&#xff0c;傳統的人工檢測方法已經不能滿足檢測需求&#xff0c;從人工檢測到自動化監測已是必然趨勢。 a. 從檢測方式看 人工檢測需要耗費大量的精力&#xff0c;從擺放檢測工具到…

VMware Workstation Pro 17下載安裝

注冊賬號 進入下載地址&#xff1a;Free Downloads - Support Portal - Broadcom support portal - https://support.broadcom.com/ 會讓注冊賬號&#xff0c;注冊一個就行 在右上角 下載 地址&#xff1a;Free Downloads - Support Portal - Broadcom support portal - ht…

SpringBoot 3.x集成阿里云OSS:文件上傳 斷點續傳 權限控制

SpringBoot 3.x集成阿里云OSS&#xff1a;文件上傳&#xff0f;斷點續傳&#xff0f;權限控制Spring Boot 3.x 集成阿里云 OSS 終極指南一、環境準備與依賴配置1. 添加阿里云 OSS SDK 依賴2. 配置 OSS 連接參數二、基礎文件上傳服務1. OSS 客戶端配置2. 文件上傳服務三、斷點續…

牛客周賽 Round 100

A小紅的雙排列沒什么好說的 直接 1 1 2 2 3 3 4 4……#define _CRT_SECURE_NO_WARNINGS #include<stdio.h> #include<iostream> #include<bits/stdc.h> #define ll long long using namespace std; int n; int main(){ios::sync_with_stdio(false); …

【Dv3Admin】菜單管理集成阿里巴巴自定義矢量圖標庫

圖標選擇是后臺管理系統中高頻功能。相比用 Element UI、Ant Design 等自帶的 icon 集&#xff0c;阿里巴巴 iconfont.cn 支持上傳和管理自定義圖標&#xff0c;并生成矢量字體&#xff0c;便于統一維護和擴展。 本文目標是支持自定義 iconfont 圖標的展示和選擇&#xff0c;并…

NO.7數據結構樹|線索二叉樹|樹森林二叉樹轉化|樹森林遍歷|并查集|二叉排序樹|平衡二叉樹|哈夫曼樹|哈夫曼編碼

線索二叉樹 線索二叉樹的基本概念 為了解決無法直接找到該結點在某種遍歷序列中的前驅和后繼結點的問題&#xff0c; 出現了線索二叉樹。 一個二叉樹通過如下的方法“穿起來” &#xff1a; 所有原本為空的右(孩子)指針改為指向該節點在某種遍歷序列中的后繼&#xff0c; 所有原…

R語言基礎| 基本圖形繪制(條形圖、堆積圖、分組圖、填充條形圖、均值條形圖)

目錄 一、前言 二、條形圖 1. 簡單的條形圖 2.堆積、分組和填充條形圖(柱狀圖) &#xff08;1&#xff09;堆積圖&#xff0c;對Improved進行堆積&#xff0c;注意position“stack” &#xff08;2&#xff09;分組圖&#xff0c;對Improved進行分組&#xff0c;注意posit…

SegNet:一種用于圖像分割的深度卷積編碼器解碼器架構

教程/講解視頻點擊文末名片1、什么是語義分割&#xff0c;什么是FCN 我們提出了一種新穎且實用的深度全卷積神經網絡架構&#xff0c;用于語義像素級分割&#xff0c;命名為SegNet。 語義分割是指為圖像中的每個像素分配一個類別標簽&#xff08;如道路、天空、汽車&#xff09…

PyTorch 數據加載全攻略:從自定義數據集到模型訓練

目錄 一、為什么需要數據加載器&#xff1f; 二、自定義 Dataset 類 1. 核心方法解析 2. 代碼實現 三、快速上手&#xff1a;TensorDataset 1. 代碼示例 2. 適用場景 四、DataLoader&#xff1a;批量加載數據的利器 1. 核心參數說明 2. 代碼示例 五、實戰&#xff1…

Python--plist文件的讀取

Python練習&#xff1a;讀取Apple Plist文件 Plist文件簡介 ??定義??&#xff1a;Apple公司創建的基于XML結構的文件格式??特點??&#xff1a;采用XML語法組織數據&#xff0c;可存儲鍵值對、數組等結構化信息文件擴展名??&#xff1a;.plist應用場景: ??iOS系統:?…

JAVA幾個注解記錄

在Java中&#xff0c;Data、AllArgsConstructor和NoArgsConstructor是Lombok庫提供的注解&#xff0c;用于自動生成Java類中的樣板代碼&#xff08;如getter、setter、構造函數等&#xff09;&#xff0c;從而減少冗余代碼&#xff0c;提高開發效率。以下是它們的詳細功能和使用…

js對象簡介、內置對象

對象、內置對象 jarringslee 對象 對象&#xff08;object&#xff09;是js的一種引用數據類型&#xff0c;是一種無序的數據集合“ul”&#xff08;類比于數組&#xff0c;有序的數據集合“ol”&#xff09;。 基本上等于結構體。 對象的聲明 //基本方法 let 對象名 {聲…

【工程篇】07:如何打包conda環境并拷貝到另一臺服務器上

這是一份以名為 qwen2.5-vl 的 Conda 環境為例的詳細操作手冊&#xff0c;指導您如何將其打包并遷移至另一臺服務器。操作手冊&#xff1a;遷移 Conda 環境 qwen2.5-vl 至新服務器 本文檔將提供兩種有效的方法來遷移您的 qwen2.5-vl 環境。請根據您的具體需求和服務器條件選擇最…

rustdesk遠控電腦替代todesk,平替向日葵等軟件

rustdesk網頁端遠控電腦docker run --restart always \ --privileged \ -p 9000:9000 \ -p 21114:21114 \ -p 21115:21115 \ -p 21116:21116 \ -p 21116:21116/udp \ -p 21117:21117 \ -p 21118:21118 \ -p 21119:21119 \ -e KEYj8muHpzr2HK00zm9D94b1UFkaJ1bEiWsyA1qxb1nOA \ …

板凳-------Mysql cookbook學習 (十二--------1)

第9章 存儲例程&#xff0c;觸發器和計劃事件 326 9.0 概述 326 9.1 創建復合語句對象 329 mysql> -- 恢復默認分隔符 mysql> DELIMITER ; mysql>mysql> DROP FUNCTION IF EXISTS avg_mail_size; Query OK, 0 rows affected (0.02 sec)mysql> DELIMITER $$ mysq…