亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C++表達式new與delete知識詳解

 更新時間:2016年05月14日 16:20:13   作者:wqli  
這篇文章主要為大家詳細(xì)介紹了C++表達式new與delete知識點,學(xué)習(xí)如何動態(tài)創(chuàng)建對象,動態(tài)創(chuàng)建的對象與一般對象的區(qū)別,動態(tài)創(chuàng)建的對象的初始化以及釋放動態(tài)分配的內(nèi)存等知識點,感興趣的朋友可以參考一下

在C++中,new表達式用于動態(tài)創(chuàng)建對象,即在堆(自由存儲區(qū))空間上為對象分配內(nèi)存,而程序員也要小心的使用這些申請來的內(nèi)存空間,當(dāng)不再使用時應(yīng)該調(diào)用delete表達式來釋放該存儲空間并且將指針置零。 本文學(xué)習(xí)了如何動態(tài)創(chuàng)建對象,動態(tài)創(chuàng)建的對象與一般對象的區(qū)別,動態(tài)創(chuàng)建的對象的初始化以及釋放動態(tài)分配的內(nèi)存等知識點。

C++中分配的內(nèi)存大致有三類:靜態(tài)存儲區(qū),棧內(nèi)存和堆內(nèi)存

其中,靜態(tài)存儲區(qū)是在程序編譯階段就已經(jīng)分配好的,用于全局變量,static變量等;堆棧是比較常用的對象存儲方式。

new和delete是C++中程序員申請和撤銷堆內(nèi)存的唯一方式(據(jù)我所知)。

1. 創(chuàng)建對象

無論是定義內(nèi)置類型變量還是類類型變量,都必須指定其對應(yīng)的數(shù)據(jù)類型和名字。

動態(tài)創(chuàng)建對象只需指定其數(shù)據(jù)類型,而不必為該對象命名。

也就是說,變量定義得到的對象可以通過額外定義的指針或其對象名本身來訪問,而動態(tài)創(chuàng)建方式得到的對象只能通過返回的指針來訪問,因為該對象并沒有名字。

2. 初始化

a. 可以在變量定義或是動態(tài)創(chuàng)建的階段初始化對象,例如:

int i(222); // definition, initialized
int *pi = new int(222) // dynamically, initialized

b. 在這還要說一說變量定義時的默認(rèn)初始化問題:

內(nèi)置類型的變量初始化與其定義位置有關(guān):函數(shù)外的自動初始化為零,而函數(shù)內(nèi)的沒有初始化。對于類類型的變量,程序總是會調(diào)用默認(rèn)構(gòu)造函數(shù)來初始化,這個默認(rèn)構(gòu)造函數(shù)可以是系統(tǒng)自動生成的,也可以是程序員定義的。如果沒有默認(rèn)構(gòu)造函數(shù),那么該類型變量的定義也就不能采用默認(rèn)初始化方式,必須提供顯式的初始化式。

動態(tài)創(chuàng)建的對象默認(rèn)初始化與函數(shù)內(nèi)變量定義的初始化方式相同,例如:

string *ps = new string; // initialized to empty string
int *pi = new int; // pi points to an uninitialized int

c. 動態(tài)創(chuàng)建對象的值初始化(value-initialize)

這種值初始化不能用于變量定義,而只能用于動態(tài)創(chuàng)建方式。

string *ps2 = new string(); // initialized to empty string
int *pi = new int(); // pi points to an int value-initialized to 0

值初始化表明程序員想要做初始化,但并未提供特定的初值。實際上,對于提供了默認(rèn)構(gòu)造函數(shù)的類類型(如string),沒有必要對其對象進行值初始化:無論程序是明確地不初始化還是要求進行值初始化,都會自動調(diào)用其默認(rèn)構(gòu)造函數(shù)初始化該對象。值初始化真正有用的是對于內(nèi)置類型。另外需要注意的一點是,值初始化的 () 語法必須置于類型名后面,而不是變量名后面,否則得到的是一個函數(shù)聲明,如下:

int myValue(); // not value-initialized int variable, but a function named myValue
int *myPtr = new int(); // correct! a value-initialized int object

對于沒有默認(rèn)構(gòu)造函數(shù)的類類型,無論是變量定義還是動態(tài)創(chuàng)建對象都必須采用顯式初始化。

3. 撤銷對象

動態(tài)創(chuàng)建的對象用完后,程序員必須顯式地將該對象占用的內(nèi)存返還給自由存儲區(qū)。C++提供了delete表達式釋放指針?biāo)赶虻牡刂房臻g。

delete myPtr;
上述語句釋放 myPtr 指向的 int 型對象所占用的內(nèi)存。

C++沒有明確定義如何釋放指向不是用 new 分配的內(nèi)存地址的指針。此外,編譯器通常不能斷定一個指針究竟指向什么類型的對象,因此如果調(diào)用delete 企圖釋放指向棧內(nèi)存地址的指針時,編譯器并不會報錯,但請盡量不要依賴于該未定義的行為。

在C++中 delete 一個零值指針是合法且安全的,但實際上毫無意義。

懸垂指針(dangling pointer)

刪除了指針?biāo)赶虻膶ο蠛螅撝羔樧兂蓱掖怪羔?。懸垂指針指向曾?jīng)存放對象的內(nèi)存,但該對象實際已經(jīng)不存在了。懸垂指針往往導(dǎo)致程序錯誤,并且很難檢測。因此,在調(diào)用delete 釋放指針?biāo)笇ο髢?nèi)存后應(yīng)該立刻將指針置零。

4. Const 對象的動態(tài)分配和回收

const 對象,必須在定義階段或動態(tài)創(chuàng)建階段進行初始化,并且初始化之后其值不能再修改。

與其他 const 對象的地址一樣,由于 new 返回的地址上存放的是 const 對象,因此該地址只能賦給指向 const 的指針。

// allocate and initialize a const object
const int *pci = new const int(222); // initialize to 222
const int *pci2 = new const int();  // initialize to 0

盡管程序員不能改變 const 對象的值,但可撤銷對象本身。

delete pci; // ok: delete a const object
三種常見的程序錯誤都與動態(tài)內(nèi)存分配相關(guān):

1、刪除動態(tài)分配內(nèi)存失敗,稱為內(nèi)存泄漏(memory leak)
2、讀寫已刪除的對象
3、對同一個內(nèi)存空間使用兩次 delete 表達式。當(dāng)兩個指針指向同一個動態(tài)創(chuàng)建的對象,刪除時就會發(fā)生錯誤。第二個指針的 delete 操作往往會破壞自由存儲區(qū)。
個人實踐部分:

#include <iostream>
#include <cstring>
using namespace std;

int main(void)
{
  string str = "hello str";
  // strNew points to dynamically allocated,
  // initialized to empty string
  string *strNew = new string;
  
  cout<<"str object address: "<<&str<<endl;
  cout<<"strNew pointer itself address: "<<&strNew<<endl;
  cout<<"strNew pointer to address: "<<strNew<<endl;
  
  // assignment
  *strNew = "hello strNew";
  cout<<"strNew pointer to address: "<<strNew<<endl;
  
  // free memory
  delete strNew;
  cout<<"strNew pointer to address: "<<strNew<<endl;
  strNew = NULL;
  
  // point to other object
  strNew = &str;
  cout<<"strNew pointer to address: "<<strNew<<endl;
  
  
  const int cvalue(10);
  // iptr points to a const int object
  const int *iptr = new const int(222);
  cout<<"iptr value: "<<*iptr<<endl;
  delete iptr;
  iptr = NULL;
  
  iptr = &cvalue;
  cout<<"iptr value: "<<*iptr<<endl;
  
  return 0;
}


一次運行的結(jié)果如下:

str object address: 0x28ff24
strNew pointer itself address: 0x28ff20
strNew pointer to address: 0x602f70
strNew pointer to address: 0x602f70
strNew pointer to address: 0x602f70
strNew pointer to address: 0x28ff24
iptr value: 222
iptr value: 10

程序中間將原來指向 new 創(chuàng)建的對象的指針重定向到一般的變量,可以看到指針存放地址的改變。另外需要注意,在釋放 new 對象之前不要將指針重新指向某個其他對象,這樣會導(dǎo)致原來動態(tài)創(chuàng)建的對象沒有指針指向它,無法釋放內(nèi)存空間。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助。

相關(guān)文章

  • QT實現(xiàn)秒表項目

    QT實現(xiàn)秒表項目

    這篇文章主要為大家詳細(xì)介紹了QT實現(xiàn)秒表項目,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-08-08
  • C++ for循環(huán)與nullptr的小知識點分享

    C++ for循環(huán)與nullptr的小知識點分享

    這篇文章主要是來和大家介紹一些C++中的小知識點,本文分享的是for循環(huán)與nullptr,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起了解一下
    2023-05-05
  • C語言中結(jié)構(gòu)體struct編寫的一些要點解析

    C語言中結(jié)構(gòu)體struct編寫的一些要點解析

    這篇文章主要介紹了C語言中結(jié)構(gòu)體struct編寫的一些要點解析,談到了結(jié)構(gòu)體的聲明和指針指向等重要知識點,需要的朋友可以參考下
    2016-04-04
  • Visual?Studio?2022?配置?PCL?1.12.1?的問題小結(jié)

    Visual?Studio?2022?配置?PCL?1.12.1?的問題小結(jié)

    這篇文章主要介紹了Visual?Studio?2022?配置?PCL?1.12.1?的經(jīng)驗總結(jié)分享,本文通過圖文實例相結(jié)合給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-08-08
  • vs2019安裝及簡單處理技巧(超詳細(xì))

    vs2019安裝及簡單處理技巧(超詳細(xì))

    這篇文章主要介紹了vs2019安裝及簡單處理方法,本文是一篇非常詳細(xì)的教程,通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-06-06
  • C語言之整數(shù)劃分問題(遞歸法)實例代碼

    C語言之整數(shù)劃分問題(遞歸法)實例代碼

    這篇文章主要介紹了C語言之整數(shù)劃分問題(遞歸法)實例代碼的相關(guān)資料,需要的朋友可以參考下
    2017-02-02
  • C++深入探究引用的使用

    C++深入探究引用的使用

    引用是C++一個很重要的特性,顧名思義是某一個變量或?qū)ο蟮膭e名,對引用的操作與對其所綁定的變量或?qū)ο蟮牟僮魍耆葍r,這篇文章主要給大家總結(jié)介紹了C++中引用的相關(guān)知識點,需要的朋友可以參考下
    2022-05-05
  • c語言可變參數(shù)實現(xiàn)示例

    c語言可變參數(shù)實現(xiàn)示例

    這篇文章主要介紹了c語言可變參數(shù)實現(xiàn)示例,需要的朋友可以參考下
    2014-04-04
  • C++編寫LINUX守護進程的實現(xiàn)代碼

    C++編寫LINUX守護進程的實現(xiàn)代碼

    這篇文章主要介紹了如何使用C++實現(xiàn)LINUX守護進程,文中代碼非常詳細(xì),供大家學(xué)習(xí)參考,感興趣的小伙伴可以了解下
    2020-06-06
  • C語言詳解如何實現(xiàn)帶頭雙向循環(huán)鏈表

    C語言詳解如何實現(xiàn)帶頭雙向循環(huán)鏈表

    帶頭雙向循環(huán)鏈表:結(jié)構(gòu)最復(fù)雜,一般用在單獨存儲數(shù)據(jù)。實際中使用的鏈表數(shù)據(jù)結(jié)構(gòu),都是帶頭雙向循環(huán)鏈表。另外這個結(jié)構(gòu)雖然結(jié)構(gòu)復(fù)雜,但是使用代碼實現(xiàn)以后會發(fā)現(xiàn)結(jié)構(gòu)會帶來很多優(yōu)勢,實現(xiàn)反而簡單
    2022-04-04

最新評論