1 运算符优先级
就跟加减乘除一样,在程序语言计算里面也有一个先后次序问题。只不过复杂性要比我们简单的加减乘除要复杂得多,里面不管是我们熟悉的数学运算,还是逻辑运算,比较运算。而这些运算也正是计算机可以解决各种错综复杂问题的来源之一。我们可以做一个总结:算术运算符 > 关系运算符 > 逻辑运算符 > 赋值运算符。
2 运算符优先级统计表
上表中可以总结出如下规律:
- 结合方向只有三个是从右往左,其余都是从左往右。
- 所有双目运算符中只有赋值运算符的结合方向是从右往左。
- 另外两个从右往左结合的运算符也很好记,因为它们很特殊:一个是单目运算符,一个是三目运算符。
- C语言中有且只有一个三目运算符。
- 逗号运算符的优先级最低,要记住。
- 此外要记住,对于优先级:算术运算符 > 关系运算符 > 逻辑运算符 > 赋值运算符。逻辑运算符中“逻辑非 !”除外。
3 一些例子
3.1 (*++p)与(*p++)
结论:(*++p)整体上是从右往左执行的,两者的优先级是相等的。两者的区别就在++p与p++的区别。i++ 返回原来的值,++i 返回加1后的值。所以来说:对( *++p)来说:是先运算++运算符然后再进行++操作,而 p++则是等p运算完,才去进行计算后面的++。
int main()
{
char charArray1[] = { 'H','e','l','l','o' };
char *p = charArray1;
//测试 *p++ 与*++p的执行结果
cout << "(**p++=)" << *p++ << endl;
cout << "*++p=" << *++p << endl;
system("pause");
return 0;
}
结果:
(**p++=)H
*++p=l
4 补充说明
4.1 左值跟右值
左值是对应内存中有确定存储地址的对象的表达式的值,而右值是所有不是左值的表达式的值。一般来说,左值是可以放到赋值符号左边的变量。但能否被赋值不是区分左值与右值的依据。比如C++的const左值是不可赋值的;而作为临时对象的右值可能允许被赋值。左值与右值的根本区别在于是否允许取地址&运算符获得对应的内存地址。
int i = 0;
int *p1 = &(++i); //正确
int *p2 = &(i++); //错误
++i = 1; //正确
i++ = 5; //错误
4.2 『i++ 不能作为左值,而++i 可以』
// 前缀形式:
int& int::operator++() //这里返回的是一个引用形式,就是说函数返回值也可以作为一个左值使用
{//函数本身无参,意味着是在自身空间内增加1的
*this += 1; // 增加
return *this; // 取回值
}
//后缀形式:
const int int::operator++(int) //函数返回值是一个非左值型的,与前缀形式的差别所在。
{//函数带参,说明有另外的空间开辟
int oldValue = *this; // 取回值
++(*this); // 增加
return oldValue; // 返回被取回的值
}
5参考资料
[1].http://c.biancheng.net/view/161.html
[2]. 添加链接在程序开发中,++i 与 i++的区别在哪里?描述
[3].为什么(i++)不能做左值,而(++i)可以