基础
- 左值和右值
- 当一个对象被用作右值的时候,用的是对象的值(内容);当对象被用作左值时,用的是对象的身份(在内存中的位置)。
- 左值可以位于赋值语句的左侧,右值则不能;
- 有几种熟悉的运算符用到左值:
- 赋值运算符(以一个左值作为其左侧运算对象,得到一个左值)
- 取地址符(作用于一个左值,返回一个右值)
- 内置解引用运算符、下标运算符、迭代器解引用运算符等(结果返回左值)
- 内置类型和迭代器的递增递减运算符作用于左值运算符(--i前置版本返回左值)
- (-m)/n和m/(-n)都等于-(m/n),m%(-n)等于m%n,(-m)%n等于-(m%n)。
递增和递减运算符
- 递增运算符和递减运算符有两种形式:前置版本和后置版本。前置版本将对象本身作为左值返回,而后置版本则将对象原始值的副本作为右值返回。
- 建议:除非必须,否则不用递增递减运算符的后置版本。
- 在一条语句中混用解引用和递增运算符
- *iter++相当于*(iter++):解引用运算符的运算对象是iter未增加之前的值
sizeof运算符
- sizeof运算符返回一条表达式或一个类型名字所占的字节数。
- sizeof运算符的结果部分依赖于其作用的类型:
- 对char或者类型为char的表达式:返回结果为1
- 对引用类型:返回为所被引用对象所占空间的大小
- 对指针:返回指针本身所占空间大小
- 对解引用指针:得到其所指对象空间大小
- 对数组:获取数组整个空间大小
类型转换
显式转换
- 一个命名的强制类型转换具有如下形式:
- cast-name<type>(expression);
- 其中type是转换的目标类型,expression是要转换的值。如果type是引用类型,则结果是左值。cast-name是static_cast、dynamic_cast、const_cast和reinterpret_cast中的一种。static_cast、const_cast比较常用。
- static_cast
- 任何具有明确定义的类型转换,只要不包含底层const,都可以使用static_cast。
int i = 10, j = 2;
double slope = static_cast<double>(i)/j;
- 当需要把一个较大的算术类型赋值给较小的类型时,编译器会给出警告信息;如果我们使用了static_cast执行了显式类型转换后,就不会有警告信息了。
- 我们还可以用static_cast召回存在于void*中的值
void *p = &d;
double *dp = static_cast<double*>(p);
- const_cast
- const_cast只能改变运算对象的底层const。
- 对于将常量对象转化为非常量的行为,我们一般称其为“去掉const性质”,一旦我们去掉了某个对象的const性质,那么就可以执行对对象的写操作了。如果对象本身不是一个常量,使用强制类型转换获得写权限是合法行为,如果对象本身是一个常量,再使用const_cast执行写操作就会产生未定义的后果。
- 只有const_cast只能改变表达式的常量属性;不能使用const_cast改变表达式的类型。
const char *pc;
char *p = const_cast<char*>(pc);//正确,但通过p写pc的值是未定义的行为
char *q = static_cast<char*>(pc);//错误,static_cast不能改变底层const
static_cast<string>(pc);//正确
const_cast<string>(cp);//错误,const_cast只改变常量属性