網站首頁 編程語言 正文
??清歡無別事,我在等風也等你
??前言
C++對象特性里的拷貝構造函數有更深入的含義,而且面試曾經問過關于拷貝的析構問題,那么今天就好好解析一下深淺拷貝的問題;還有初始化列表的形式,這個在給對象屬性初始化的時候非常方便,建議大家熟練掌握,話不多說,開始正文。
??目錄
深淺拷貝區別
代碼解釋
特別注意
內存圖解釋
初始化列表
語法
具體實現
??總結
深淺拷貝區別
上一節簡單提了編譯器會默認給我們提供值拷貝構造函數,結果是新的對象會擁有和傳入對象一樣的屬性,由編譯器提供的拷貝構造被稱為淺拷貝構造,而由我們自己編寫的不同于編譯器提供的拷貝構造函數就叫深拷貝構造了,舉個典型的例子說明。
代碼解釋
#include
using namespace std;
//深淺拷貝問題,存在經典的坑,面試考過
class Person
{
public:
Person(int age,int height)
{
m_age = age;
m_Height = new int(height);
cout << "Person 的有參構造函數調用" << endl;
}
//自己寫不同于編譯器的拷貝構造函數屬于深拷貝
Person(const Person& p)
{
cout << "拷貝構造函數調用" << endl;
m_age = p.m_age;
//m_Height = p.m_Height;
/*編譯器默認執行上行代碼
新開辟的地址相同,會導致調用析構函數時違法操作,無法訪問內存*/
m_Height =new int(*p.m_Height);
}
~Person()
{
//析構代碼,將堆區開辟的數據做釋放操作
if (m_Height != NULL)
{
delete m_Height;
m_Height = NULL;
}
cout << "~Person 的析構構造函數調用"<
創建Person類,設置m_age和指針類型*m_Height為私有屬性;依次對Person類設置有參構造和拷貝構造函數以及析構函數;前面析構函數一直沒有什么作用,其實它是用來清理對象的,析構函數會在程序結束前自動調用,這時候就可以使用delete清理掉;
特別注意
Person(const Person& p)
{
cout << "拷貝構造函數調用" << endl;
m_age = p.m_age;
//m_Height = p.m_Height;
/*編譯器默認執行上行代碼
新開辟的地址相同,會導致調用析構函數時違法操作,無法訪問內存*/
m_Height =new int(*p.m_Height);
}
~Person()
{
//析構代碼,將堆區開辟的數據做釋放操作
if (m_Height != NULL)
{
delete m_Height;
m_Height = NULL;
}
cout << "~Person 的析構構造函數調用"<
這里不能使用編譯器提供的淺拷貝,如果直接使用m_Height=p.m_Height,毫無疑問這兩個屬性地址相同,那么在調用析構函數的時候,p1先釋放內存,這時候雖然有一個NULL判斷,但是此塊內存已經被刪除,再次訪問都會提示錯誤,這是很危險的,所以我們需要用深拷貝解決重復刪除的問題。使用m_Height=new int(*p.m_Height) 語句給身高屬性重新開辟空間,這樣在調用析構的時候各自清理各自的屬性,就解決了這個淺拷貝帶來的重復清理問題。
內存圖解釋
上面是淺拷貝的p1、p2對象的內存示意圖,兩次析構會重復當問0x00011地址,但是當這個地址被刪除后,是不允許再次訪問的。
利用我們設置的深拷貝構造后,地址不一樣,各自刪除各自的地址,解決問題
初始化列表
初始化列表用來給屬性初始化
語法
普通構造函數+?:?+ 類屬性(變量或常量)+ {}
具體實現
class Person
{
public:
Person() :m_age(20), m_sex(1), m_height(180){}
Person(int a, int b, int c) :m_age(a), m_sex(b), m_height(c){}
int m_age;
int m_sex;
int m_height;
};
int main()
{
Person p1;
Person p2(10, 20, 30);
cout << "年齡為:" << p1.m_age;
cout << "性別為:" << p1.m_sex;
cout << "身高為:" << p1.m_height<
主函數中p1調用無參構造函數,各屬性初始化為屬性()括號里面的值;p2調用有參構造函數,將實參10,20,30分別傳給a,b,c,然后a的值傳給m_age,b的值傳給m_sex;c的值傳給m_height;直接來看結果:
??總結
C++對象特性的深淺拷貝和初始化列表到這里就分享完了,那麼意味著對象的初始化和清理就結束了,到后面會講一下靜態成員和類對象作為類成員的一個案例,用于鞏固這部分知識點,希望下篇文章也可以得到你們的青睞,下次再見!
原文鏈接:https://blog.csdn.net/m0_58618795/article/details/124912579
相關推薦
- 2022-09-18 C++?STL反向迭代器的實現_C 語言
- 2022-08-22 Python利用字典和列表實現學生信息管理系統_python
- 2022-03-26 C++約瑟夫環問題詳解_C 語言
- 2022-05-17 bat批處理腳本中文亂碼的解決_DOS/BAT
- 2022-09-08 深入了解Go語言的基本語法與常用函數_Golang
- 2023-04-04 numpy中的norm()函數求范數實例_python
- 2022-12-05 Linux中的grep?-v、-e、-E用法小結_linux shell
- 2022-03-14 ffmpeg開發讀取目錄列表
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支