可以将C ++中的Assignment Operator虚拟化。为什么需要它?我们也可以使其他运营商虚拟吗?
Answers:
不需要将赋值运算符设为虚拟。
下面的讨论是关于的operator=
,但它也适用于接受了所讨论类型的任何运算符重载,以及接受了所讨论类型的任何函数。
下面的讨论表明,在寻找匹配函数签名方面,虚拟关键字不知道参数的继承。在最后一个示例中,它显示了在处理继承类型时如何正确处理分配。
虚函数不知道参数的继承:
要使虚拟发挥作用,功能的签名必须相同。因此,即使在以下示例中,将operator =设为虚拟,该调用也永远不会充当D中的虚拟函数,因为operator =的参数和返回值不同。
该功能B::operator=(const B& right)
与D::operator=(const D& right)
100%完全不同,被视为2个不同的功能。
class B
{
public:
virtual B& operator=(const B& right)
{
x = right.x;
return *this;
}
int x;
};
class D : public B
{
public:
virtual D& operator=(const D& right)
{
x = right.x;
y = right.y;
return *this;
}
int y;
};
默认值,并具有2个重载运算符:
虽然可以定义一个虚函数,以便在将D分配给类型B的变量时为D设置默认值。即使您的B变量确实是存储在B的引用中的D,也不会。D::operator=(const D& right)
功能。
在以下情况下,将使用存储在2个B引用中的2个D对象的赋值...使用D::operator=(const B& right)
覆盖。
//Use same B as above
class D : public B
{
public:
virtual D& operator=(const D& right)
{
x = right.x;
y = right.y;
return *this;
}
virtual B& operator=(const B& right)
{
x = right.x;
y = 13;//Default value
return *this;
}
int y;
};
int main(int argc, char **argv)
{
D d1;
B &b1 = d1;
d1.x = 99;
d1.y = 100;
printf("d1.x d1.y %i %i\n", d1.x, d1.y);
D d2;
B &b2 = d2;
b2 = b1;
printf("d2.x d2.y %i %i\n", d2.x, d2.y);
return 0;
}
印刷品:
d1.x d1.y 99 100
d2.x d2.y 99 13
这表明D::operator=(const D& right)
从未使用过。
如果不使用virtual关键字,B::operator=(const B& right)
您将获得与上述相同的结果,但是y的值不会被初始化。即它将使用B::operator=(const B& right)
RTTI将这一切捆绑在一起的最后一步:
您可以使用RTTI正确处理传入类型的虚函数。这是弄清楚在处理可能的继承类型时如何正确处理分配的最后一个难题。
virtual B& operator=(const B& right)
{
const D *pD = dynamic_cast<const D*>(&right);
if(pD)
{
x = pD->x;
y = pD->y;
}
else
{
x = right.x;
y = 13;//default value
}
return *this;
}
这取决于操作员。
使赋值运算符虚拟化的目的是使您能够重写它以复制更多字段。
因此,如果您有一个Base&,并且实际上有一个Derived&作为动态类型,并且Derived具有更多字段,则将复制正确的内容。
但是,存在这样的风险:您的LHS是派生的,而RHS是基本的,因此当虚拟运算符在派生中运行时,您的参数不是派生的,并且您将无法从中获取字段。
这是一个很好的讨论:http : //icu-project.org/docs/papers/cpp_report/the_assignment_operator_revisited.html
布莱恩·邦迪写道:
RTTI将这一切捆绑在一起的最后一步:
您可以使用RTTI正确处理传入类型的虚函数。这是弄清楚在处理可能的继承类型时如何正确处理分配的最后一个难题。
virtual B& operator=(const B& right) { const D *pD = dynamic_cast<const D*>(&right); if(pD) { x = pD->x; y = pD->y; } else { x = right.x; y = 13;//default value } return *this; }
我想在此解决方案中添加一些说明。让赋值运算符声明与上述相同有三个问题。
编译器生成一个赋值运算符,该赋值运算符带有一个const D&参数,该参数不是虚拟的,并且没有执行您可能认为的操作。
第二个问题是返回类型,您正在返回对派生实例的基本引用。无论如何,代码可能不会有太大的问题。仍然最好相应地返回引用。
第三个问题,派生类型赋值运算符不会调用基类赋值运算符(如果您要复制私有字段,该怎么办?),将赋值运算符声明为virtual不会使编译器为您生成一个。这是没有赋值运算符的至少两个重载来获得所需结果的副作用。
考虑基类(与我引用的帖子中的基类相同):
class B
{
public:
virtual B& operator=(const B& right)
{
x = right.x;
return *this;
}
int x;
};
以下代码完善了我引用的RTTI解决方案:
class D : public B{
public:
// The virtual keyword is optional here because this
// method has already been declared virtual in B class
/* virtual */ const D& operator =(const B& b){
// Copy fields for base class
B::operator =(b);
try{
const D& d = dynamic_cast<const D&>(b);
// Copy D fields
y = d.y;
}
catch (std::bad_cast){
// Set default values or do nothing
}
return *this;
}
// Overload the assignment operator
// It is required to have the virtual keyword because
// you are defining a new method. Even if other methods
// with the same name are declared virtual it doesn't
// make this one virtual.
virtual const D& operator =(const D& d){
// Copy fields from B
B::operator =(d);
// Copy D fields
y = d.y;
return *this;
}
int y;
};
这似乎是一个完整的解决方案,但事实并非如此。这不是一个完整的解决方案,因为当您从D派生时,您将需要1个运算符=接受const B&,1个运算符=接受const D&,以及一个需要const D2&的运营商。结论是显而易见的,运算符=()重载的数量等于超类的数量+ 1。
考虑到D2继承了D,让我们看一下两个继承的operator =()方法的外观。
class D2 : public D{
/* virtual */ const D2& operator =(const B& b){
D::operator =(b); // Maybe it's a D instance referenced by a B reference.
try{
const D2& d2 = dynamic_cast<const D2&>(b);
// Copy D2 stuff
}
catch (std::bad_cast){
// Set defaults or do nothing
}
return *this;
}
/* virtual */ const D2& operator =(const D& d){
D::operator =(d);
try{
const D2& d2 = dynamic_cast<const D2&>(d);
// Copy D2 stuff
}
catch (std::bad_cast){
// Set defaults or do nothing
}
return *this;
}
};
显然,运算符=(const D2&)仅复制字段,想象一下它在那里。我们可以注意到继承的运算符=()重载中的模式。可悲的是,我们无法定义可以处理这种模式的虚拟模板方法,我们需要多次复制和粘贴相同的代码才能获得完整的多态赋值运算符,这是我所看到的唯一解决方案。也适用于其他二进制运算符。
如评论中所述,使生活变得更轻松的最少方法是定义最顶层的超类赋值运算符=(),并从所有其他超类运算符=()方法中调用它。同样,在复制字段时,可以定义_copy方法。
class B{
public:
// _copy() not required for base class
virtual const B& operator =(const B& b){
x = b.x;
return *this;
}
int x;
};
// Copy method usage
class D1 : public B{
private:
void _copy(const D1& d1){
y = d1.y;
}
public:
/* virtual */ const D1& operator =(const B& b){
B::operator =(b);
try{
_copy(dynamic_cast<const D1&>(b));
}
catch (std::bad_cast){
// Set defaults or do nothing.
}
return *this;
}
virtual const D1& operator =(const D1& d1){
B::operator =(d1);
_copy(d1);
return *this;
}
int y;
};
class D2 : public D1{
private:
void _copy(const D2& d2){
z = d2.z;
}
public:
// Top-most superclass operator = definition
/* virtual */ const D2& operator =(const B& b){
D1::operator =(b);
try{
_copy(dynamic_cast<const D2&>(b));
}
catch (std::bad_cast){
// Set defaults or do nothing
}
return *this;
}
// Same body for other superclass arguments
/* virtual */ const D2& operator =(const D1& d1){
// Conversion to superclass reference
// should not throw exception.
// Call base operator() overload.
return D2::operator =(dynamic_cast<const B&>(d1));
}
// The current class operator =()
virtual const D2& operator =(const D2& d2){
D1::operator =(d2);
_copy(d2);
return *this;
}
int z;
};
不需要设置默认值的方法,因为它只会收到一个调用(在基本运算符=()重载中)。在一个地方完成复制字段时的更改,并且所有operator =()重载都将受到影响并具有其预期的目的。
感谢sehe的建议。
在以下情况下使用虚拟分配:
//code snippet
Class Base;
Class Child :public Base;
Child obj1 , obj2;
Base *ptr1 , *ptr2;
ptr1= &obj1;
ptr2= &obj2 ;
//Virtual Function prototypes:
Base& operator=(const Base& obj);
Child& operator=(const Child& obj);
情况1:obj1 = obj2;
在这种虚拟概念中,我们operator=
在Child
上课时不会扮演任何角色。
情况2&3:* ptr1 = obj2;
* ptr1 = * ptr2;
此处的分配将不符合预期。原因operator=
是在Base
类上被调用。
可以使用以下任
一方法进行纠正:1)铸造
dynamic_cast<Child&>(*ptr1) = obj2; // *(dynamic_cast<Child*>(ptr1))=obj2;`
dynamic_cast<Child&>(*ptr1) = dynamic_cast<Child&>(*ptr2)`
2)虚拟概念
现在,仅使用便virtual Base& operator=(const Base& obj)
无济于事,因为签名Child
和Base
for中的签名不同operator=
。
我们需要添加Base& operator=(const Base& obj)
Child类及其常规Child& operator=(const Child& obj)
定义。重要的是要包含以后的定义,因为在没有默认赋值运算符的情况下,它将被调用。(obj1=obj2
可能不会给出期望的结果)
Base& operator=(const Base& obj)
{
return operator=(dynamic_cast<Child&>(const_cast<Base&>(obj)));
}
情况4:obj1 = * ptr2;
在这种情况下,编译器会operator=(Base& obj)
定义Child
为operator=
叫上孩子。但由于它不存在,且Base
类型不能被晋升为child
含蓄,它会通过错误。(铸件需要像obj1=dynamic_cast<Child&>(*ptr1);
)
如果我们根据case2&3实施,则将解决此情况。
可以看出,在使用基类指针/引用进行分配的情况下,虚拟分配使调用更加优雅。
我们也可以使其他运营商虚拟吗? 是
dynamic_cast<const Child &>(obj)
代替有意义dynamic_cast<Child&>(const_cast<Base&>(obj))
吗?
short
至int
...)。