关闭

漫谈C++重载运算符

发表于:2010-2-21 10:14

字体: | 上一篇 | 下一篇 | 我要投稿

 作者:mynote    来源:51Testing软件测试网采编

#
DotNet

  1.前置运算符和后置运算符,左值和右值。

  其实很久以来一直都没有怎么搞清楚左值和右值的区别,只知道左值可以放在等号的左边,也可以放在等号的右边,但是右值却只能放在等号的右边,然后形成一个大概直观的印象,知道怎么样做才不出错而已。不过今天看看C++,却发现有了点新的体会。

  对于表达式a--=5;这样一个表达式,明显是错误的,究其原因,是因为执行等号左边的自减表达式之后,显示取得a的值,然后才是进行自减操作,所以最终的结果是一个右值,而且就是a的值(变化前的),于是对于赋值语句 b = a--,自然不会弄错了。

  而对于--a=5;这样一个表达式,则明显就不同了。首先执行的是自减操作,然后返回的是a的值(变化后的),于是自然就可以将返回的a值再次赋值了。也就是说前置的运算符是左值表达式,而后置的则是右值表达式。于是,我想到了很久前的一个想法:++a++ = 5;当然,行家肯定一看就知道是错误的,可是错误的地方需要改正啊。于是我想到了几种方法,就是通过加括号的方法来实现。

  1)++(a++) = 5; 其实通过错误信息就可以看到原因了:error: non-lvalue in increment ,明显说明在自增运算符中的那个表达式不是左值,所以不能自增运算了。哦,对啊,a++是右值,于是的话再次进行++的操作就是错误的,因为++操作需要的表达式是左值表达式。于是此处的方法行不通。

  2)(++a)++ = 5;此处也不行,错误信息就是: error: non-lvalue in assignment,通过前面的表述就知道了,其实++a是左值表达式,那么后面的那个++操作就是一个错误了,赋值的时候,等号的左边不是左值表达式,因为括号外的是一个后置的++运算符。于是,就得到了错误。

  通过上面的这些折腾,对于++++a等等变态的表达式,相信也就有更深刻的体会了。

  2.重载前置++和后置++的时候的不同

  相信很多人都做过这种操作了,对于后置的++,需要多一个形参int来说明,但是从上面的分析,还需要注意的是两者的左值和右值的区别,重载的目的,也不过是为了更加符合用户的习惯,于是,我们对于前置的++,自然要返回的是左值表达式,而对于后置的++,需要返回的则是左值表达式。当然,具体的来说说吧。

  前置++:如果自定义了一个类A(含有一个数据成员x),那么要重载它的前置++运算符,根据习惯,首先就是对它的数据成员++,然后再返回它的引用,这样,才是左值表达式,于是下面的代码: A& A::operator++(){ ++x; return *this ; }这样的方法,自然很好的表达了前置的作用,先运算,在返回值。

  后置++:如果同上面的例子,要定义后置++的话,那么需要注意的是它的右值性,此时是返回引用还是类类型呢?思考了很久,一般来说,返回的是引用的,都是左值表达式,而如果此处返回的是引用,那么就可能在后面的代码中出现问题。

  于是出现了下面的这样的代码:

A A::operator++(int )
{
 return A(this->x++);
}

  其实我觉得此句甚好,既表达了后置的意思,又成功的返回了值。后置的含义,也就是先返回之后再进行自增操作,于是,此处调用的是默认的拷贝构造函数来实现返回一个临时的变量。

21/212>
《2023软件测试行业现状调查报告》独家发布~

关注51Testing

联系我们

快捷面板 站点地图 联系我们 广告服务 关于我们 站长统计 发展历程

法律顾问:上海兰迪律师事务所 项棋律师
版权所有 上海博为峰软件技术股份有限公司 Copyright©51testing.com 2003-2024
投诉及意见反馈:webmaster@51testing.com; 业务联系:service@51testing.com 021-64471599-8017

沪ICP备05003035号

沪公网安备 31010102002173号