++i 和 i++ 的区别和实现
++i 和 i++ 的区别和实现_c+的博客-CSDN博客_++i i++ https://blog.csdn.net/qnavy123/article/details/79482999
在程序开发中,++i 与 i++的区别在哪里? - 知乎 https://www.zhihu.com/question/19811087
一、++i 和 i++ 的区别:
1. i++返回的是i的值, ++i返回的是i+1的值;
2. i++不能用作左值, ++i 可以用作左值。
二、左右值的根本区别是什么?
答:左右值的根本区别是: 是否允许用取地址运算符&获取相应的内存地址。
三、 i++ 和++i的实现
1. ++i 实现:
int& int::operator++()
{
*this +=1;
return *this;
}
2. i++ 实现:
const int int::operator(int)
{
int oldValue = *this;
++(*this);
return oldValue;
}
从效率上来看, ++i 效率高于 i++
链接:https://www.zhihu.com/question/19811087/answer/80210083
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
i++ 与 ++i 的主要区别有两个:
1、 i++ 返回原来的值,++i 返回加1后的值。
2、 i++ 不能作为左值,而++i 可以。
毫无疑问大家都知道第一点(不清楚的看下下面的实现代码就了然了),我们重点说下第二点。
首先解释下什么是左值(以下两段引用自中文维基百科『右值引用』词条)。
左值是对应内存中有确定存储地址的对象的表达式的值,而右值是所有不是左值的表达式的值。
一般来说,左值是可以放到赋值符号左边的变量。但
能否被赋值不是区分左值与右值的依据。比如,C++的const左值是不可赋值的;而作为临时对象的右值可能允许被赋值。左值与右值的根本区别在于是否允许取地址&运算符获得对应的内存地址。
比如,
int i = 0;
int *p1 = &(++i); //正确
int *p2 = &(i++); //错误
++i = 1; //正确
i++ = 5; //错误
那么为什么『i++ 不能作为左值,而++i 可以』?
看它们各自的实现就一目了然了:
以下代码来自博客:
为什么(i++)不能做左值,而(++i)可以// 前缀形式:
int& int::operator++() //这里返回的是一个引用形式,就是说函数返回值也可以作为一个左值使用
{//函数本身无参,意味着是在自身空间内增加1的
*this += 1; // 增加
return *this; // 取回值
}
//后缀形式:
const int int::operator++(int) //函数返回值是一个非左值型的,与前缀形式的差别所在。
{//函数带参,说明有另外的空间开辟
int oldValue = *this; // 取回值
++(*this); // 增加
return oldValue; // 返回被取回的值
}
如上所示,i++ 最后返回的是一个临时变量,而临时变量是右值。
算下来已经很久以前了,大概有那么几个星期了吧。看到了帖子上有人问为什么下面的语句的对错原因
int i = 0;
int *ip = &(i++); //错误
int *ip = &(++i); //正确
看到上面的语句,当时以为两个都是是对的啊,可是当我用VS2010的编译器试了试之后,才发现第一个取地址确实是错误的语句,具体的提示是:error C2102: “&”要求左值,而第二个取地址是正确的。
也就是说取地址符"&"作用的对象必须是一个左值,左值的概念是什么呢?看一下百度对于C/C++中左值的定义:
左值:
C/C++语言中可以放在赋值符号左边的变量,即具有对应的可以由用户访问的存储单元,并且能够由用户去改变其值的量。左值表示存储在计算机内存的对象,而不是常量或计算的结果。或者说左值是代表一个内存地址值,并且通过这个内存地址,就可以对内存进行读并且写(主要是能写)操作;这也就是为什么左值可以被赋值的原因了。相对应的还有右值:当一个符号或者常量放在操作符右边的时候,计算机就读取他们的“右值”,也就是其代表的真实值。简单来说就是,左值相当于地址值,右值相当于数据值
看完左值的定义就不难理解为什么取地址运算符需要作用在一个左值对象上了。可为什么i++与++i有如此的区别呢?
原因是:i++不是存储在x中的值,他们的具体函数实现请看下面:
// 前缀形式:
int& int::operator++() //这里返回的是一个引用形式,就是说函数返回值也可以作为一个左值使用
{//函数本身无参,意味着是在自身空间内增加1的
*this += 1; // 增加
return *this; // 取回值
}
//后缀形式:
const int int::operator++(int) //函数返回值是一个非左值型的,与前缀形式的差别所在。
{//函数带参,说明有另外的空间开辟
int oldValue = *this; // 取回值
++(*this); // 增加
return oldValue; // 返回被取回的值
}