當前位置:學問君>人在職場>IT認證>

計算機二級C++輔導知識點積累

學問君 人氣:2.45W

c++語言還允許使用一種以“”開頭的特殊形式的字元常量。這種字元常量稱爲轉義字元,用來表示一些不可顯示的或有特殊意義的字元。以下是關於計算機二級C++輔導知識點,歡迎大家參考!

計算機二級C++輔導知識點積累

一、析構函數

前面的一些例子都沒有說明析構函數,這是因爲所用到的類在結束時不需要做特別的清理工作。下面的程序給出了一新的Date類,其中包括一個字元串指針,用來表示月份。

#include iostream.h

#include string.h

class Date

{

int mo,da,yr;

char *month;

public:

Date(int m=0, int d=0, int y=0);

~Date();

void display() const;

};

Date::Date(int m,int d,int y)

{

static char *mos[] =

{

January,February,March,April,May,June,

July,August,September,October,November,December

};

mo=m; da=d; yr=y;

if(m!=0)

{

month=new char[strlen(mos[m-1])+1];

strcpy(month, mos[m-1]);

}

else month = 0;

}

Date::~Date()

{

delete [] month;

}

void Date::display() const

{

if(month!=0) cout< }

int main()

{

Date birthday(8,11,1979);

lay();

return 0;

}

在Date對象的構造函數中,首先用new運算符爲字元串month動態分配了內存,然後從內部數組中把月份的名字拷貝給字元串指針month。

析構函數在刪除month指針時,可能會出現一些問題。當然從這個程序本身來看,沒什麼麻煩;但是從設計一個類的角度來看,當Date類用於賦值時,就會出現問題。假設上面的main()修改爲“

int main()

{

Date birthday(8,11,1979);

Date today;

today=birthday;

lay();

return 0;

}

這會生成一個名爲today的空的Date型變量,並且把birthday值賦給它。如果不特別通知編譯器,它會簡單的認爲類的賦值就是成員對成員的拷貝。在上面的程序中,變量birthday有一個字元型指針month,並且在構造函數裏用new運算符初始化過了。當birthday離開其作用域時,析構函數會調用delete運算符來釋放內存。但同時,當today離開它的作用域時,析構函數同樣會對它進行釋放操作,而today裏的month指針是birthday裏的month指針的一個拷貝。析構函數對同一指針進行了兩次刪除操作,這會帶來不可預知的後果

如果假設today是一個外部變量,而birthday是一個自變量。當birthday離開其作用域時,就已經把對象today裏的month指針刪除了。顯然這也是不正確的`。

再假設有兩個初始化的Date變量,把其中一個的值賦值給另一個:

Date birthday(8,11,1979);

Date today(12,29,2003);

today=birthday;

問題就更復雜了,當這兩個變量離開作用域時,birthday中的month的值已經透過賦值傳遞給了today。而today中構造函數用new運算符給month的值卻因爲賦值被覆蓋了。這樣,birthday中的month被刪除了兩次,而today中month卻沒有被刪除掉。

二、重載賦值運算符

爲了解決上面的問題,我們應該寫一個特殊的賦值運算符函數來處理這類問題。當需要爲同一個類的兩個對象相互賦值時,就可以重載運算符函數。這個方法可以解決類的賦值和指針的釋放。

下面的程序中,類中的賦值函數用new運算符從堆中分配了一個不同的指針,該指針獲取賦值對象中相應的值,然後拷貝給接受賦值的對象。

在類中重載賦值運算符的格式如下:

void operator = (const Date&)

後面我們回加以改進。目前,重載的運算符函數的返回類型爲void。它是類總的成員函數,在本程序紅,是Date類的成員函數。它的函數名始終是operator =,參數也始終是同一個類的對象的引用。參數表示的是源對象,即賦值數據的提供者。重載函數的運算符作爲目標對象的成員函數來使用。

#include iostream.h

#include string.h

class Date

{

int mo,da,yr;

char *month;

public:

Date(int m=0, int d=0, int y=0);

~Date();

void operator=(const Date&);

void display() const;

};

Date::Date(int m, int d, int y)

{

static char *mos[] =

{

January,February,March,April,May,June,

July,August,September,October,November,December

};

mo = m; da = d; yr = y;

if (m != 0)

{

month = new char[strlen(mos[m-1])+1];

strcpy(month, mos[m-1]);

}

else month = 0;

}

Date::~Date()

{

delete [] month;

}

void Date::display() const

{

if (month!=0) cout< char name[25];

cin >> name;

if (strncmp(name, end, 3) == 0) break;

ListEntry* list = new ListEntry(name);

if (prev != 0) prev->AddEntry(*list);

prev = list;

}

while (prev != 0)

{

prev->display();

ListEntry* hold = prev;

prev = prev->PrevEntry();

delete hold;

}

return 0;

}

程序執行時,會提示輸入一串姓名,當輸入完畢後,鍵入end,然後程序會逆序顯示剛纔輸入的所有姓名。

程序中ListEntry類含有一個字元串和一個指向前一個表項的指針。構造函數從對中獲取內存分配給字元串,並把字元串的內容拷貝到內存,然後置連結指針爲NULL。析構函數將釋放字元串所佔用的內存。

成員函數PrevEntry()返回指向鏈表前一個表項的指針。另一個成員函數顯示當前的表項內容。

成員函數AddEntry(),它把this指針拷貝給參數的preventry指針,即把當前表項的地址賦值給下一個表項的連結指針,從而構造了一個鏈表。它並沒有改變調用它的listEntry對象的內容,只是把該對象的地址賦給函數的參數所引用的那個ListEntry對象的preventry指針,儘管該函數不會修改對象的數據,但它並不是常量型。這是因爲,它拷貝對象的地址this指針的內容給一個非長常量對象,而編譯器回認爲這個非常量對象就有可能透過拷貝得到的地址去修改當前對象的數據,因此AddEntry()函數在聲明時不需要用const.