自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

new[]和delete[]一定要配對(duì)使用嗎?

開發(fā) 后端
在平時(shí)資料中,我們常看到:new和delete,new[]和delete[]一定要配對(duì)使用!有時(shí)候不配對(duì)使用也不會(huì)出現(xiàn)問(wèn)題。也許你也是只知其然,不知其所以然,然而我也有點(diǎn)懵了。

[[438453]]

 在平時(shí)資料中,我們常看到:new和delete,new[]和delete[]一定要配對(duì)使用!

也有人說(shuō):有時(shí)候不配對(duì)使用也不會(huì)出現(xiàn)問(wèn)題。也許你也是只知其然,不知其所以然,然而我也有點(diǎn)懵了_(¦3」∠)_

那就研究下這個(gè)問(wèn)題:

首先,看下這段配對(duì)使用代碼: 

  1. #include <stdlib.h>  
  2. #include <iostream>  
  3. using namespace std;  
  4. class inner {  
  5.    public:  
  6.     inner() { cout << "Constructing" << endl; }  
  7.     ~inner() { cout << "Destructing" << endl; }  
  8. };  
  9. int main(int argc, char *argv[]) {  
  10.     inner *p = new inner();  
  11.     inner *pa = new inner[2];  
  12.     delete p;  
  13.     delete []pa;  
  14.     return 0;  
  15.  
  16. 程序輸出:  
  17. Constructing  
  18. Constructing  
  19. Constructing  
  20. Destructing  
  21. Destructing  
  22. Destructing 

因?yàn)閚ew[]會(huì)創(chuàng)建一個(gè)數(shù)組,一個(gè)對(duì)象數(shù)組需要一定的空間大小,假設(shè)一個(gè)對(duì)象需要N字節(jié)大小,K個(gè)對(duì)象的數(shù)組就需要K*N個(gè)空間來(lái)構(gòu)造對(duì)象數(shù)組,但是在delete[]時(shí)候,如何知道數(shù)組的長(zhǎng)度呢?

所以new[]會(huì)在K*N個(gè)空間的基礎(chǔ)上,頭部多申請(qǐng)4個(gè)字節(jié),用于存儲(chǔ)數(shù)組長(zhǎng)度,這樣delete[]時(shí)候才知道對(duì)象數(shù)組的大小,才會(huì)相應(yīng)調(diào)用K次析構(gòu)函數(shù),并且釋放K*N+4大小的內(nèi)存。

這是我們平時(shí)編程中經(jīng)常配對(duì)使用的情況,如果不配對(duì)使用呢?

new[]與delete結(jié)對(duì)使用 

  1. #include <stdlib.h>  
  2. #include <iostream>  
  3. using namespace std;  
  4. class inner {  
  5.    public:  
  6.     inner() { cout << "Constructing" << endl; }  
  7.     ~inner() { cout << "Destructing" << endl; }  
  8. };  
  9. int main(int argc, char *argv[]) {  
  10.     inner *p = new inner[2];  
  11.     delete p;  
  12.     return 0;  
  13.  
  14. 程序輸出:  
  15. Constructing  
  16. Constructing  
  17. Destructing  
  18. munmap_chunk(): invalid pointer  
  19. Aborted (core dumped) 

這里發(fā)現(xiàn):程序掛掉了。

并且,只調(diào)用了一次析構(gòu)函數(shù),為什么呢?

因?yàn)槲覀兪褂昧薲elete,delete不同于delete[],它認(rèn)為這只是一個(gè)對(duì)象占用的空間,不是對(duì)象數(shù)組,不會(huì)訪問(wèn)前4個(gè)字節(jié)獲取長(zhǎng)度,所以只調(diào)用了一次析構(gòu)函數(shù)。而且,最后釋放內(nèi)存的時(shí)候只釋放了起始地址為A的內(nèi)存。然而這不是這一整塊內(nèi)存的起始地址,整塊內(nèi)存的起始地址應(yīng)該是A-4,釋放內(nèi)存如果不從內(nèi)存起始地址操作就會(huì)出現(xiàn)斷錯(cuò)誤,所以導(dǎo)致程序掛掉。

關(guān)于內(nèi)存知識(shí)可以看我以前的文章:

10張圖22段代碼,萬(wàn)字長(zhǎng)文帶你搞懂虛擬內(nèi)存模型和malloc內(nèi)部原理

new和delete[]結(jié)對(duì)使用 

  1. #include <stdlib.h>  
  2. #include <iostream>  
  3. using namespace std;  
  4. class inner {  
  5.    public:  
  6.     inner() { cout << "Constructing" << endl; }  
  7.     ~inner() { cout << "Destructing" << endl; }  
  8. };  
  9. int main(int argc, char *argv[]) {  
  10.     inner *p = new inner();  
  11.     delete []p;  
  12.     return 0; 
  13. 程序輸出:  
  14. Constructing  
  15. Destructing  
  16. Destructing  
  17. Destructing  
  18. Destructing  
  19. Destructing  
  20. Destructing  
  21. ...  
  22. Destructing  
  23. free(): invalid pointer  
  24. Aborted (core dumped) 

這里調(diào)用了不定次數(shù)的析構(gòu)函數(shù),并且掛掉,是因?yàn)樵趎ew時(shí)候沒(méi)有多申請(qǐng)4個(gè)字節(jié)存儲(chǔ)長(zhǎng)度,而delete[]時(shí)候還會(huì)向前找4個(gè)字節(jié)獲取長(zhǎng)度,這4個(gè)字節(jié)是未定義的,所以調(diào)用了不固定次數(shù)的析構(gòu)函數(shù),釋放內(nèi)存的時(shí)候也釋放了起始地址為A-4的內(nèi)存,而正常的起始地址應(yīng)該是A,所以程序掛掉。

什么時(shí)候可以不配對(duì)使用?

我們?cè)賮?lái)看一段代碼: 

  1. #include <stdlib.h>  
  2. #include <iostream>  
  3. using namespace std;  
  4. int main() {  
  5.     int *pint = new int(5);  
  6.     delete[] pint;  
  7.     int *pinta = new int[4];  
  8.     delete pinta;  
  9.     cout << "success" << endl 
  10.     return 0;  
  11.  
  12. 程序輸出:  
  13. success 

這段代碼即使不配對(duì)使用也會(huì)正常運(yùn)行,這是為什么呢,因?yàn)閕nt是內(nèi)置類型,new[]和delete[]在配合int使用時(shí)知道int是內(nèi)置類型,不需要析構(gòu)函數(shù),所以也就不需要多4個(gè)字節(jié)來(lái)存放數(shù)組長(zhǎng)度,只需要直接操作內(nèi)存即可。 

 

責(zé)任編輯:龐桂玉 來(lái)源: C語(yǔ)言與C++編程
相關(guān)推薦

2015-09-21 09:02:39

java數(shù)組

2015-09-16 13:11:23

Java數(shù)組初始化

2021-01-05 05:27:32

業(yè)務(wù)Leader團(tuán)隊(duì)

2022-04-24 09:54:24

ProxyReflect前端

2017-10-16 12:52:51

2010-11-19 16:02:42

IT族

2017-05-05 09:26:33

2021-03-29 22:58:34

大數(shù)據(jù)Java編程語(yǔ)言

2021-03-16 15:49:30

架構(gòu)運(yùn)維技術(shù)

2024-05-10 08:10:05

Spring虛擬線程JDK

2021-03-05 11:02:14

iOS 14.5蘋果更新

2011-11-09 14:54:50

Linux操作系統(tǒng)

2022-05-30 07:36:07

Python腳本函數(shù)

2022-08-31 22:50:13

JavaScript函數(shù)JSON

2019-12-31 09:43:54

微服務(wù)JavaDocker

2022-04-27 07:21:06

HTTPAPI系統(tǒng)

2012-07-18 16:34:35

企業(yè)級(jí)服務(wù)器SSDPCIe

2022-06-13 09:26:41

Promise前端代碼

2021-10-29 06:56:15

Python腳本解釋器

2023-09-26 07:43:22

工具骨架屏頁(yè)面
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)