分享
 
 
 

正确的重载operator+

王朝other·作者佚名  2006-01-08
窄屏简体版  字體: |||超大  

正确的重载operator+

(Overloading Operator + the Right Way)

By Danny Kalev, C++ Pro 翻译:周尚延beejoy

为了得到一串链表或者完成两个对象的累加或合并,经常要重载诸如string,date,complex或file等user-define类型的Operator+。然而,就设计而言,正确地重载operator+面临许多挑战。下面的段落中,我将介绍为用户定义类型选择正确的重载operator+的一般性的策略。

考虑下面的表达式:

int x=4+2;

内建的+运算符把两个类型相同的操作数相加然后返回右值6传递给x,因此我们可以说:内建的运算符’+‘是基于二进制的、对称的并且是可交换的操作符,它返回一个与操作数具有相同类型的值。这是一个规则,当你为user-defined类型重载一个运算符时,应该维持相应内建运算符的特性。

为user-defined类型重载operator+是个很普通的事情。不过,由于C++提供了多种实现方式,导致在设计时很容易造成错误,从而对利用标准库组件写出的代码的正确性、执行性能和兼容性造成不利影响。

分析并内建操作符的特性去重载与之相应的user-define运算符。

第一步:选择成员函数或是非成员函数?

我们可以象使用类的成员函数一样使用+、-或是==等二进制操作符,例如:

class String

{

public:

bool operator==(const String & s); // compare *this and s

};

然而,这种实现方式值得怀疑。在这种情况下,作为内建操作符的副本,重载后的操作符违背了内建操作符具有的对称性,它的两个操作符的类型分别是'const String* const'(隐含的this 参数)和'const String &',这将使得一些STL运算和某一些容器不能正确的处理这些对象。

另一种反其道而行之的方法是,把重载的operator+定义为外部函数,两个参数的类型相同,如下:

String operator + (const String & s1, const String s2);

在这种情况下,类String要定义运算符的重载函数为友元函数:

class String

{

public:

friend String operator+(const String& s1,const String&s2);

};

第二步:返回值,进退两难的选择

正如前面所述,内建的operator+返回同它的操作符类型相同的右值。不过在caller的堆栈中运行对象将是低效率的,特别是在处理大对象的时候。能不能用返回一个指针或是一个传引用来代替?在这里不行。我们要求变量和返回值的类型要相同,返回指针破坏了这一特性。更糟糕的是,这使得我们不能使用多重连续表达式:

String s1,s2,s3;

String res;

res=s1+s2+s3; //返回不合法的String*

尽管有一个解决办法,也就是另外定义operator+的不同的重载函数, 但这个解决方案没有考虑到另一种情况:当返回的指针指向的是一个动态分配的对象,如果调用者没有清除这个返回的指针,就会导致内存泄漏。所以很明显的,返回String*是个糟糕的想法。

那么可不可以返回String&?同样的,返回的引用必须绑定到一个合法的String。接下来,既然动态分配对象不行,那么第二次我们何不选择返回一个对局部静态变量的引用?是的,静态对象可以解决内存泄漏的问题。不过这仍然有些不可靠。比如在一个多线程程序中,两个线程可能同时调用operator+,这就会导致对string对象的误用。此外,因为静态对象保留上一次调用时的状态,所以我们必须在每次调用operator+前清空这个静态string对象。这样我们得到结论,返回堆栈中的值仍然是最安全并且最简单的解决方案。

第三步:实现

到目前为止一切顺利。在前面我们采用了外部友元函数并且采用by value返回值方式。那么现在该是我们实现重载opertor+函数的时候了。然而又一次,我们面临两种选择:

选择一:Intrusive Concatenation模型

在Intrusive Concatenation模型中,operator+测量函数参数的长度,复制字符串到一个足够地大的缓冲区并且返回结果 ( 为简洁的缘故,我忽略了不常见的情况例如堆过载):

// Option 1: intrusive concatenation

String operator + (const String & s1, const String s2)

{

int len=s1.size()+s2.size();

String result;

result.buf=new char[len+1];

strcpy(result.buf, s1.buf);

strcat(result.buf, s2.buf); // concatenate second string

return result;

}

这个实现能够较好的工作,不过还有更简洁更灵活的基于重载的operator+=的解决方案。

选择二:使用运算符+=

我们使用operator+=来,从而避免对private数据的存取、内存分配以及c函数的使用。

// Option 2: using operator +=

String operator + (const String & s1, const String s2)

{

String result=s1;

result+=s2;

return result;

}

这个实现方法比较容易,而且不会存取string类的非公有数据。并且,我们甚至不需要把它声明为友元函数。operator+=的重载实现留给读者作为练习。

作为重载operator+运算符的最广泛的例子,我们讨论的焦点集中在了自己编写的一个string类。当然,你可以把我介绍的方法和策略应用到任意的user-defined类型上。正确的重载operator+基于以下原则:

●运算符对称性

●返回值与操作符类型相同

●采用by value返回方式

●采用基于重载+=运算符的实行方式

(完)

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有