C、C++ 中的 i++与++i异同

 

  1. 在C和C++中,i++返回原来的值,++i返回自增后的值;

  2. 在C++中,++i为左值,i++为右值;

  3. 在C中,i++和++i都为非左值。

对于1,应该都很清楚,讨论下2和3。对于2,首先看看维基百科对左值右值的阐述[1]

在C++11提出右值引用之前,C++03及更早的C++标准中,表达式的“值分类”(value categories)属性为左值或右值。[参 2]左值是对应(refer to)内存中有确定存储地址的对象的表达式的值,而右值是所有不是左值的表达式的值。因而,右值可以是字面量[注 1]、临时对象[注 2]等表达式。能否被赋值不是区分C++左值与右值的依据。C++的const左值是不可赋值的;而作为临时对象的右值可能允许被赋值。左值与右值的根本区别在于是否允许取地址&运算符获得对应的内存地址

以下是一段C++代码,使用g++编译

int i = 0;
int *p1 = &(++i); //正确
int *p2 = &(i++); //错误

++i = 1; //正确
i++ = 5; //错误

在C++中,『i++』是右值,不允许取地址&运算符获得对应的内存地址,实际上,i++ 返回一个临时变量,这一点可看它具体的实现[3]

// 前缀形式:
int& int::operator++() //这里返回的是一个引用形式,就是说函数返回值也可以作为一个左值使用
{//函数本身无参,意味着是在自身空间内增加1的
  *this += 1;  // 增加
  return *this;  // 取回值
}

//后缀形式:
const int int::operator++(int) //函数返回值是一个非左值型的,与前缀形式的差别所在。
{//函数带参,说明有另外的空间开辟
  int oldValue = *this;  // 取回值
  ++(*this);  // 增加
  return oldValue;  // 返回被取回的值,是个临时变量
}

 

对于3,由于在C中,一个表达式要么是左值要么是非左值,没有右值说法,直到C++时,右值才被正名[2]。所以以下代码在C中都是编译不过的,

// using gcc
int i = 0;
int *p1 = &(++i); //错误
int *p2 = &(i++); //错误

++i = 1; //错误
i++ = 5; //错误

 

参考:

[1] https://zh.wikipedia.org/wiki/%E5%8F%B3%E5%80%BC%E5%BC%95%E7%94%A8

[2] http://www.cnblogs.com/dejavu/archive/2012/09/02/2667640.html

[3] http://blog.csdn.net/zlhy_/article/details/8349300

posted @ 2016-07-16 17:24  加菲猫007  阅读(427)  评论(0编辑  收藏  举报