细谈C/C++中的前置++和后置++
2011-02-27 14:22
253 查看
++运算符是大家都再熟悉不过的东西了,无论是在for循环里,还是在一些赋值语句中,都有++的身影。这里我和大家分享一下我对++的理解,若有不足之处欢迎批评指正啊。首先,看这样一段代码:
虽然加号多的让人有点想吐,但还是有规律可循的。错误中出现了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; 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; }
相关文章推荐
- C++再学习系列:前置++与后置++
- c++ 前置++与后置++的区别
- [C++] C++的运算符重载(+、-、前置--、后置--,前置++,后置++、==)
- [C++] C++的运算符重载(+、-、前置--、后置--,前置++,后置++、==)
- c++基础3:关于前置++和后置++的那么点纠结
- c++ 前置++与后置++的区别
- C++中前置++和后置++的区别
- C++中前置++与后置++运算符重载
- 对于C++中的前置++和后置++的看法
- C++再学习系列:前置++与后置++
- C++中的前置++和后置++
- [C++] 前置++和后置++的区别
- C++回顾之前置++、后置++、不等号!及赋值运算符重载
- C++回顾之前置++、后置++、不等号!及赋值运算符重载
- C++重载++ 如何区分“++”重载的前置与后置
- 项目二对象前置++,前置--;后置++,后置--
- 前置++,后置++与运算符计算顺序问题
- C++ 后置运算符重载 180P
- C++学习笔记--前置、后置操作符
- java与C++中的后置自增操作符++