C/C++細節問題(bug調試)以動態綁定代碼來示例
在程序界,有句話是,程序好些,bug難調。
很多bug並不一點是邏輯錯誤,而是一些細節問題。
我們分析下三個細節錯誤引起的無法運行問題:
//#define M_PI 3. 14159265358979323846 3.14中間有個空格
\t 寫成了\ t 有個空格
首先,創建了一個新類型CFigure。我們想創建一些具體的圖形(如,三角形、正方形或者圓),以及計算這些圖形周長和面積的方法。但是,我們並不知道具體的圖形是什麽類型,所以無法用方法直接計算圖形的這些特性。這就是要把CFigure類創建為抽象類的原因。抽象類是至少聲明了一個虛方法的類,該虛方法沒有實現,且其原型後面有=0。以這種方式聲明的函數叫做純虛函數。抽象類不能有對象,但是可以有繼承類。因此可以實例化抽象類的指針和引用,然後從CFigure類派生出CTriangle、CSquare和CCircle類,分別表示三角形、正方形和圓形。
我們要實例化這些對象的類型,所以在這些派生類中,實現了FigureType方法、Circumference方法和Area方法。雖然這3個類中的方法名都相同,但是它們的實現不同,這與覆蓋類似但含義不同。
如何理解?在本例的main函數中,聲明了一個數組,內含3個CFigure類型的指針。作為指向基類的指針或引用,它們一定可以指向該基類的任何派生類。因此,可以創建一個CTriangle類型的對象,並設置CFigure類型的指針指向它,
CFigure* figures[3];
同理,用下面的代碼可以設置其他圖形:
figures[0] = new CTriangle(2.1,3.2,4.3); figures[1] = new CSquare(5.4,6.6); figures[2] = new CCircle(8.9);
現在,考慮下面的代碼:
for (int i = 0; i < 3; i++) { cout << "Figure type:\t" << figures[i]->FigureType(); cout<< "\nCircumference:\t" << figures[i]->Circumference(); cout<< "\nArea:\t\t" << figures[i]->Area(); cout<< endl << endl; }
編譯器將使用C++的動態綁定(dynamicbinding)特性,確定圖形指針具體指向哪個類型的對象,調用合適的虛方法。只有把方法聲明為虛方法,且通過指針或引用訪問才能使用動態綁定。
全部代碼如下,直接可以編譯運行:
#include "stdafx.h" #include < iostream> #include <tchar.h> #define M_PI 3.14159265358979323846 using namespace std; //#define M_PI 3. 14159265358979323846 3.14中間有個空格 class CFigure //base class { public: virtual char* FigureType() = 0; virtual double Circumference() = 0; virtual double Area() = 0; virtual ~CFigure() { } }; class CTriangle : public CFigure //Derived Class { public: CTriangle() { a = b = c = 0; } CTriangle(double a, double b, double c) : a(a), b(b), c(c) { } virtual char* FigureType() { return "Triangle"; } virtual double Circumference() { return a + b + c; } virtual double Area() { double S = Circumference() / 2; return sqrt(S * (S - a) * (S - b) * (S - c)); } private: double a, b, c; }; class CSquare : public CFigure //Derived Class { public: CSquare() { a = 0.0; b = 0.0; } CSquare(double a, double b) : a(a), b(b) { } virtual char* FigureType() { return "Square"; } virtual double Circumference() { return 2 * a + 2 * b; } virtual double Area() { return a * b; } private: double a, b; }; class CCircle : public CFigure //Derived Class { private: double r; public: CCircle() { r = 0; } CCircle(double r) : r(r) { } virtual char* FigureType() { return "Circle"; } virtual double Circumference() { //return 2 * r * M_PI; double res = 0.00000000000000000000000; res = 2 * r * M_PI; //res = 2 * r * 3. 141592; return res; } virtual double Area() { double res = 0.000000000000000000000; res = 2 * r * M_PI; //res = 2 * r * 3. 14159265358979323846; return res; } }; int _tmain(int argc, _TCHAR* argv[]) { CFigure* figures[3]; figures[0] = new CTriangle(2.1,3.2,4.3); figures[1] = new CSquare(5.4,6.6); figures[2] = new CCircle(8.9); for (int i = 0; i < 3; i++) { cout << "Figure type:\t" << figures[i]->FigureType(); cout<< "\nCircumference:\t" << figures[i]->Circumference(); cout<< "\nArea:\t\t" << figures[i]->Area(); cout<< endl << endl; } getchar(); return 0; }
運行結果:
本文出自 “享受科技帶來的快樂” 博客,請務必保留此出處http://liam2199.blog.51cto.com/2879872/1963425
C/C++細節問題(bug調試)以動態綁定代碼來示例