- 相關(guān)推薦
計算機二級考試C++復習資料
一、析構函數
前面的一些例子都沒(méi)有說(shuō)明析構函數,這是因為所用到的類(lèi)在結束時(shí)不需要做特別的清理工作。下面的程序給出了一新的Date類(lèi),其中包括一個(gè)字符串指針,用來(lái)表示月份。
#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()
{
[] month;
}
void Date::display() const
{
if(month!=0) cout< }
int main()
{
Date birthday(8,11,1979);
birthday.display();
return 0;
}
在Date對象的構造函數中,首先用new運算符為字符串month動(dòng)態(tài)分配了內存,然后從內部數組中把月份的名字拷貝給字符串指針month。
析構函數在刪除month指針時(shí),可能會(huì )出現一些問(wèn)題。當然從這個(gè)程序本身來(lái)看,沒(méi)什么麻煩;但是從設計一個(gè)類(lèi)的角度來(lái)看,當Date類(lèi)用于賦值時(shí),就會(huì )出現問(wèn)題。假設上面的main()修改為“
int main()
{
Date birthday(8,11,1979);
Date today;
today=birthday;
birthday.display();
return 0;
}
這會(huì )生成一個(gè)名為today的空的Date型變量,并且把birthday值賦給它。如果不特別通知編譯器,它會(huì )簡(jiǎn)單的認為類(lèi)的賦值就是成員對成員的拷貝。在上面的程序中,變量birthday有一個(gè)字符型指針month,并且在構造函數里用new運算符初始化過(guò)了。當birthday離開(kāi)其作用域時(shí),析構函數會(huì )調用運算符來(lái)釋放內存。但同時(shí),當today離開(kāi)它的作用域時(shí),析構函數同樣會(huì )對它進(jìn)行釋放操作,而today里的month指針是birthday里的month指針的一個(gè)拷貝。析構函數對同一指針進(jìn)行了兩次刪除操作,這會(huì )帶來(lái)不可預知的后果。
如果假設today是一個(gè)外部變量,而birthday是一個(gè)自變量。當birthday離開(kāi)其作用域時(shí),就已經(jīng)把對象today里的month指針刪除了。顯然這也是不正確的。
再假設有兩個(gè)初始化的Date變量,把其中一個(gè)的值賦值給另一個(gè):
Date birthday(8,11,1979);
Date today(12,29,2003);
today=birthday;
問(wèn)題就更復雜了,當這兩個(gè)變量離開(kāi)作用域時(shí),birthday中的month的值已經(jīng)通過(guò)賦值傳遞給了today。而today中構造函數用new運算符給month的值卻因為賦值被覆蓋了。這樣,birthday中的month被刪除了兩次,而today中month卻沒(méi)有被刪除掉。
二、重載賦值運算符
為了解決上面的問(wèn)題,我們應該寫(xiě)一個(gè)特殊的賦值運算符函數來(lái)處理這類(lèi)問(wèn)題。當需要為同一個(gè)類(lèi)的兩個(gè)對象相互賦值時(shí),就可以重載運算符函數。這個(gè)方法可以解決類(lèi)的賦值和指針的釋放。
下面的程序中,類(lèi)中的賦值函數用new運算符從堆中分配了一個(gè)不同的指針,該指針獲取賦值對象中相應的值,然后拷貝給接受賦值的對象。
在類(lèi)中重載賦值運算符的格式如下:
void operator = (const Date&)
后面我們回加以改進(jìn)。目前,重載的運算符函數的返回類(lèi)型為void。它是類(lèi)總的成員函數,在本程序紅,是Date類(lèi)的成員函數。它的函數名始終是operator =,參數也始終是同一個(gè)類(lèi)的對象的引用。參數表示的是源對象,即賦值數據的提供者。重載函數的運算符作為目標對象的成員函數來(lái)使用。
#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()
{
[] 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();
hold;
}
return 0;
}
程序運行時(shí),會(huì )提示輸入一串姓名,當輸入完畢后,鍵入end,然后程序會(huì )逆序顯示剛才輸入的所有姓名。
程序中ListEntry類(lèi)含有一個(gè)字符串和一個(gè)指向前一個(gè)表項的指針。構造函數從對中獲取內存分配給字符串,并把字符串的內容拷貝到內存,然后置鏈接指針為NULL。析構函數將釋放字符串所占用的內存。
成員函數PrevEntry()返回指向鏈表前一個(gè)表項的指針。另一個(gè)成員函數顯示當前的表項內容。
成員函數AddEntry(),它把this指針拷貝給參數的preventry指針,即把當前表項的地址賦值給下一個(gè)表項的鏈接指針,從而構造了一個(gè)鏈表。它并沒(méi)有改變調用它的listEntry對象的內容,只是把該對象的地址賦給函數的參數所引用的那個(gè)ListEntry對象的preventry指針,盡管該函數不會(huì )修改對象的數據,但它并不是常量型。這是因為,它拷貝對象的地址this指針的內容給一個(gè)非長(cháng)常量對象,而編譯器回認為這個(gè)非常量對象就有可能通過(guò)拷貝得到的地址去修改當前對象的數據,因此AddEntry()函數在聲明時(shí)不需要用const。
【計算機二級考試C++復習資料】相關(guān)文章:
計算機二級考試科目07-17
關(guān)于計算機二級考試大綱復習07-02
計算機二級VB上機考試答題攻略07-02
計算機二級考試筆試及機試備考復習指導技巧09-27
事業(yè)單位考試復習資料求助07-10
計算機二級VFP筆試真題07-03
二級建造師考試備考03-21
二級建造師考試備考06-26
管理學(xué)原理復習資料(考試必備)08-25