網站首頁 編程語言 正文
深淺拷貝區別
上一節簡單提了編譯器會默認給我們提供值拷貝構造函數,結果是新的對象會擁有和傳入對象一樣的屬性,由編譯器提供的拷貝構造被稱為淺拷貝構造,而由我們自己編寫的不同于編譯器提供的拷貝構造函數就叫深拷貝構造了,舉個典型的例子說明。
代碼解釋
#include<iostream>
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 的析構構造函數調用"<<endl;
}
int m_age;
int* m_Height;
};
void test()
{
Person p1(20,180);
Person p2(p1);
cout << "p2.age= " << p2.m_age << " p2.height=" << *p2.m_Height << endl;
}
int main()
{
test();
system("pause");
}
創建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 的析構構造函數調用"<<endl;
}
這里不能使用編譯器提供的淺拷貝,如果直接使用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<<endl;
cout << "年齡為:" << p2.m_age;
cout << "性別為:" << p2.m_sex;
cout << "身高為:" << p2.m_height << endl;;
}
主函數中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
相關推薦
- 2023-10-17 el-table-column 表單table的后端返回時間戳的轉換
- 2023-02-12 redis加鎖的三種方式小結_Redis
- 2022-08-11 淺析k8s中各組件和kube?apiserver通信時的認證和鑒權問題_云其它
- 2022-12-10 Qt中簡單的按鈕槽函數傳遞參數方法_C 語言
- 2022-06-13 nginx?location/區別詳解_nginx
- 2023-01-05 Python中glob類的使用方法_python
- 2022-05-13 當你敲完Hello World后的第一步——C語言
- 2023-12-09 使用String.valueOf()的坑
- 最近更新
-
- 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同步修改后的遠程分支