中重载操作符的正确方法

液压机械/部件2021年09月06日

用户定义的类型,如:字符串,日期,复数,联合体以及文件常常重载二元 + 操作符以实现对象的连接,附加或合并机制。但是要正确实现 + 操作符会给设计,实现和性能带来一定的挑战。本文将概要性地介绍如何选择正确的策略来为用户定义类型重载这个操作符。

考虑如下的表达式: int x=4+2;

内建的 + 操作符有两个类型相同的操作数,相加并返回右值 6,然后被赋值给 x。我们可以断定内建的 + 是一个二元的,对称的,可交换的操作符。它产生的结果的类型与其操作数类型相同。按照这个规测,当你为某个用户定义类型重载操作符时,也应该遵循相应内建操作符的特征。

为用户定义类型重载 + 操作符是很常见的编程任务。尽管 C++ 提供了几种实现方法,但是它们容易使人产生设计上的误解,这种误解常常影响代码的正确性,性能以及与标准库组件之间的兼容性。

按照申花队今年的思路 下面我们就来分析内建操作符的特征并尝试模仿其相应的重载机制。

第一步:在成员函数和非成员函数之间选择

你可以用类成员函数的方式实现二元操作符如:+、- 以及 ==,例如:

class String

{

public:

bool operator==(const String s); // 比较 *this 和 s

};

这个方法是有问题的。相对于其内建的操作符来说,重载的操作符在这里不具有对称性;它的两个参数一个类型为:const String * const(这个参数是隐含的),另一个类型为:const String 。因此,一些 STL 算法和容器将无法正确处理这样的对象。

另外一个可选方法是把重载操作符 + 定义为一个外部(extern)函数,该函数带两个类型相同的参数:

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

这样一来,类 String 必须将该重载操作符声明为友元:

class String

{

public:

friend String operator+(const String s1,const Strings2);

};

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

如前所述,内建操作符 + 返回右值,其类型与操作数相同。但是在调用者堆栈里返回一个对象效率很低,处理大型对象时尤其如此。那么能不能返回一个指针或引用呢?答案是不行。因为返回指针破坏参数类型与返回值类型应该相同的规则。更糟的是,链接多个表达式将成为不可能:

String s1,s2,s3;

String res;

res=s1+s2+s3; // 不可能用 String* 作为返回值

虽然有一个办法可以定义额外的 + 操作符重载版本,但这个办法是我们不希望用的,因为返回的指针必须指向动态分配的对象。这样的话,如果调用者释放(delete)返回的指针失败,那么将导致内存泄漏。显然,返回 String* 不是一个好主意。

那么返回 String 好不好呢?返回的引用必须一定要是一个有效的 String。它避免了使用动态对象分配,该方法返回的是一个本地静态对象的引用。静态对象确实解决了内存泄漏问题,但这个方法的可行性仍然值得怀疑。在一个多线程应用中,两个线程可能会并发调用 + 操作符,因此造成 String 对象的混乱。而且,因为静态对象总是保留其调用前的状态,所以有必要针对每次 + 操作符的调用都清除该静态 String 对象。由此看来,在堆栈上返回结果仍然是最安全和最简单的解决方案。

查看本文来源

南宁宫颈糜烂治疗哪家好
呼和浩特哪医院治疗白癜风好
韶关治白癜风哪里最好
相关阅读
发端诗意流、半途主义的著名作家唐国明,鹅毛诗10首

引领意象迳、四人法西斯主义的著名小问道家夏氏容,大白赋10首 (提醒:本...

2023-09-16
从“狗玩的”到“人玩的”:飞盘在年轻人当中爆火,却离商业化很远

保龄球正扭转着其在社会大招集心之中的常规印象,逐渐从“犬碰的”替换成...

2023-09-15
24岁嫁进豪门,王祖贤第三者加入致使分手,62岁成为单身亿万富婆

24岁嫁进到豪门,刘嘉玲第三者重新加入导致两人,62岁沦为单身亿万富婆!...

2023-09-10
为什么佩戴笔记本电脑后能听见声音却听不清?

有些女朋友觉得戴上助哭器后就无论如何马上每个字都哭特别确实,那事实上...

2023-09-07
刘雯登七月刊封面,造型大胆前卫挑战金属故作,网友:“吉巴罗”既视故作

《嘉人》七月刊封面大片截止,刘雯大胆挑战前卫、锆自觉十足的样貌,每一...

2023-09-05
2021年工资公布!IT最赚钱,这些行业“不如意”

中所新财经5月21日电(记者 李金磊)2021年加班费数据库先于。 其中所,各地区...

2023-09-04
友情链接