您的位置:首页 > 编程语言 > C语言/C++

细谈C/C++中的前置++和后置++

2011-02-27 14:22 253 查看
++运算符是大家都再熟悉不过的东西了,无论是在for循环里,还是在一些赋值语句中,都有++的身影。这里我和大家分享一下我对++的理解,若有不足之处欢迎批评指正啊。首先,看这样一段代码:
#include <iostream>
using namespace std;

int main()
{
int i;
++++i; // All Right
//i++++; // error C2105: '++' needs l-value
//++i=10; // All Right
//i++=10; // error C2106: '=' : left operand must be l-value
//++i++; // error C2105: '++' needs l-value
return 0;
}


虽然加号多的让人有点想吐,但还是有规律可循的。错误中出现了l-value(左值),那我们先弄清楚什么是左值吧。

左值是可以写在=号左边的表达式,必须是一个非const的变量。
右值是可以写在=号右边的表达式,可以是一个常量,一个字面数字,或者说一个任意的可以求值的表达式。一个可以作为左值的表达式,同时也是可以作为右值使用的。

原来,问题的根源就是对右值进行了左值操作(赋值,++等)。对于i++++,i++是右值,所以i++++是对右值进行了++操作,所以错误。而++++i是对++i(左值)进行++操作,所以运行成功。++i=10;和i++=10;的问题就显而易见了。问题到这里还没结束,接下来我们分析一下为什么i++是右值,而++i是左值。实际上,i++等价于i=i+1,i。而i++等价于tmp=i,i=i+1,tmp,其中tmp是由编译器产生的一个临时变量。而临时变量只能作为右值使用,例如函数的返回值一般就是一个临时变量。
再谈一下++i和i++的效率问题,很显然,++i的效率要高于i++,因为i++产生了一个临时变量tmp。所以,下次我们可以优化一下我们的程序了。
例如,将for(int i=0;i<end;i++)改为for(int i=0;i<end;++i)。

对于C++重载++运算符时,也需要注意++运算符的特性。例如:

#include <iostream>
using namespace std;

class PlusPlus
{
public:
// 构造函数
PlusPlus()
{}
PlusPlus(int value):data(value)
{}
// 重载前置++
PlusPlus& operator++();
// 重载后置++
PlusPlus operator++(int);
// 重载<<运算符
friend ostream& operator << (ostream& out,const PlusPlus& obj);
int GetData();
void SetData(int value);
private:
int data;
};

PlusPlus& PlusPlus::operator++()
{
data++;
return *this;
}

PlusPlus PlusPlus::operator++(int)
{
PlusPlus tmp;
tmp.SetData(data);
data++;
return tmp;
}

int PlusPlus::GetData()
{
return data;
}

void PlusPlus::SetData(int value)
{
data=value;
}

ostream& operator << (ostream& out,PlusPlus& obj)
{
out<<"data : "<<obj.GetData();
return out;
}

int main()
{
PlusPlus obj(10);
obj++;
cout<<obj<<endl;
++obj;
cout<<obj<<endl;
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: