这个例子摘自Bruce Eckel的《C++编程思想》第14章节的"向上转型和复制构造函数"。
我也运行这个程序,通过用以下指令替换上述指令:
#include <iostream>
using namespace std;
class Parent
{
int i;
public:
Parent(int ii) : i(ii) { cout << "Parent(int ii)\n"; }
Parent(const Parent& b) : i(b.i) { cout << "Parent(const Parent&)\n"; }
Parent() : i(0) { cout << "Parent()\n"; }
friend ostream& operator<<(ostream& os, const Parent& b)
{ return os << "Parent: " << b.i << endl; }
};
class Member
{
int i;
public:
Member(int ii) : i(ii) { cout << "Member(int ii)\n"; }
Member(const Member& m) : i(m.i) { cout << "Member(const Member&)\n"; }
friend ostream& operator<<(ostream& os, const Member& m)
{ return os << "Member: " << m.i << endl; }
};
class Child : public Parent
{
int i;
Member m;
public:
Child(int ii) : Parent(ii), i(ii), m(ii) { cout << "Child(int ii)\n"; }
friend ostream& operator<<(ostream& os, const Child& c)
{ return os << (Parent&)c << c.m << "Child: " << c.i << endl; }
};
int main() {
Child c(2);
cout << "calling copy-constructor: " << endl;
Child c2 = c;
cout << "values in c2:\n" << c2;
}
关于这段代码,作者发表了以下评论:
“Child 类的 << 操作符非常有趣,因为它在其中调用了 Parent 部分的 << 操作符:通过将 Child 对象强制转换为 Parent&(如果你将其强制转换为基类对象而不是引用,通常会得到不良结果):
return os << (Parent&)c << c.m << "Child: " << c.i << endl;
我也运行这个程序,通过用以下指令替换上述指令:
return os << (Parent)c << c.m << "Child: " << c.i << endl;
程序可以顺利运行,只有一个预期的区别。现在再次调用 Parent
的拷贝构造函数将参数 c
复制到 Parent::operator<<()
中。
那么作者所说的不良结果是什么呢?
const
转换为非const
对象的强制转换。应该将(Parent&)c
改为(const Parent &)c
吗? - Ed HealPatent::operator<<()
的第二个参数是一个const Parent&
。 - Belloc