一、關(guān)于流運(yùn)算符(<<、>>)的重載問(wèn)題###
首先考察一個(gè)實(shí)部虛部為double類型的復(fù)數(shù)complex類:
class complex
{
public:
complex (double r = 0, double i = 0): re (r), im (i) { }
//……
double real () const { return re; }
double imag () const { return im; }
private:
double re, im;
//……
};
對(duì)<<操作符進(jìn)行重載,使cout能夠以“(x + yi)”的格式輸出complex對(duì)象時(shí),可以這么做:
ostream& operator<<(ostream& os, const complex& c)
{
os << "(" << c.real() << " + " << c.imag() << "i)";
return os;
}
如果不想調(diào)用real()和imag(),想直接訪問(wèn)re和im這兩個(gè)私有成員,那么我們要把operator <<定義為complex類的友元函數(shù):
class complex
{
//……
//在complex類里聲明友元
public:
friend ostream& operator<<(ostream& os, const complex& c);
//……
};
ostream& operator<<(ostream& os, const complex& c)
{
os << "(" << c.re << " + " << c.im << "i)";
return os;
}
那么,為什么不能把<<運(yùn)算符直接重載為complex類的成員函數(shù)呢?這樣不用聲明友元不也可以訪問(wèn)私有成員了嗎?
假如真的這么做了,那么我們的代碼應(yīng)該是這個(gè)樣子:
class complex
{
public:
//……
ostream& operator<<(ostream& os, const complex& c);
//……
};
但是這是成員函數(shù)呀,前面還應(yīng)該有個(gè)隱含的this指針才對(duì),<<是雙目運(yùn)算符,只能接收兩個(gè)參數(shù),所以這么寫不對(duì);
那我們重載<<的目的就是為了將complex對(duì)象的輸出格式放到cout里,所以只好改成這樣:
class complex
{
public:
//……
ostream& operator<<(ostream& os);
//……
};
//<<的實(shí)現(xiàn)
ostream& complex::operator<<(ostream& os)
{
os << "(" << c.re << " + " << c.im << "i)";
return os;
}
然后用的時(shí)候,因?yàn)槌蓡T運(yùn)算符函數(shù)固定左值為該類的對(duì)象,所以只好寫成這樣:
complex c1(1, 2);
c1 << cout;
和我們平時(shí)用的cout << c1這樣格式方向相反,非常別扭,
而且如果要實(shí)現(xiàn)連續(xù)輸出cout << c1 << c2 << ......,呃,不對(duì),現(xiàn)在的例子應(yīng)該是...... << c2 << c1 << cout,除了最右邊那個(gè)<<可以調(diào)用,其他的<<因?yàn)榻邮盏膮?shù)是complex類型,所以我們還要進(jìn)行一次重載。種種麻煩說(shuō)明,還是老老實(shí)實(shí)把<<重載寫在類外面,或者聲明友元好了。