优先级 | 运算符 | 名称或含义 | 使用形式 | 结合方向 | 说明 |
---|---|---|---|---|---|
1 | [] | 数组下标 | 数组名[常量表达式] | ||
() | 圆括号 | (表达式) 函数名(形参表) | |||
. | 成员选择(对象) | 对象.成员名 | |||
-> | 成员选择(指针) | 对象指针->成员名 | |||
2 | - | 负号运算符 | -表达式 | 右到左 | 单目运算符 |
(类型) | 强制类型转换 | (数据类型)表达式 | |||
++ | 自增运算符 | ++变量名 变量名++ | 单目运算符 | ||
– | 自减运算符 | –变量名 变量名– | 单目运算符 | ||
* | 取值运算符 | *指针变量 | 单目运算符 | ||
& | 取地址运算符 | &变量名 | 单目运算符 | ||
! | 逻辑非运算符 | !表达式 | 单目运算符 | ||
~ | 按位取反运算符 | ~表达式 | 单目运算符 | ||
sizeof | 长度运算符 | sizeof(表达式) | |||
3 | / | 除 | 表达式 / 表达式 | 双目运算符 | |
* | 乘 | 表达式*表达式 | 双目运算符 | ||
% | 余数(取模) | 整型表达式%整型表达式 | 双目运算符 | ||
4 | + | 加 | 表达式+表达式 | 双目运算符 | |
- | 减 | 表达式-表达式 | 双目运算符 | ||
5 | << | 左移 | 变量<<表达式 | 双目运算符 | |
>> | 右移 | 变量>>表达式 | 双目运算符 | ||
6 | > | 大于 | 表达式>表达式 | 双目运算符 | |
>= | 大于等于 | 表达式>=表达式 | 双目运算符 | ||
< | 小于 | 表达式<表达式 | 双目运算符 | ||
<= | 小于等于 | 表达式<=表达式 | 双目运算符 | ||
7 | == | 等于 | 表达式==表达式 | 双目运算符 | |
!= | 不等于 | 表达式!= 表达式 | 双目运算符 | ||
8 | & | 按位与 | 表达式&表达式 | 双目运算符 | |
9 | ^ | 按位异或 | 表达式^表达式 | 双目运算符 | |
10 | | | 按位或 | 表达式|表达式 | 双目运算符 | |
11 | && | 逻辑与 | 表达式&&表达式 | 双目运算符 | |
12 | || | 逻辑或 | 表达式||表达式 | 双目运算符 | |
13 | ?: | 条件运算符 | 表达式1? 表达式2: 表达式3 | 右到左 | 三目运算符 |
14 | = | 基本赋值运算符 | 变量=表达式 | 右到左 | |
/= | 除后赋值 | 变量/=表达式 | |||
*= | 乘后赋值 | 变量*=表达式 | |||
%= | 取模后赋值 | 变量%=表达式 | |||
+= | 加后赋值 | 变量+=表达式 | |||
-= | 减后赋值 | 变量-=表达式 | |||
<<= | 左移后赋值 | 变量<<=表达式 | |||
>>= | 右移后赋值 | 变量>>=表达式 | |||
&= | 按位与后赋值 | 变量&=表达式 | |||
^= | 按位异或后赋值 | 变量^=表达式 | |||
|= | 按位或后赋值 | 变量|=表达式 | |||
15 | , | 逗号运算符 | 表达式,表达式,… |
规律:
- 结合方向只有三个是从右往左, 其余都是从左往右. 其中扩展的赋值运算符的结合方向和基本赋值运算符的结合方向相同.
- 所有双目运算符中只有赋值运算符的结合方向是从右往左.
- 另外两个从右往左结合的运算符也很好记:一个是单目运算符中的负号运算符, 一个是三目运算符即条件运算符.
- C 语言中有且只有一个三目运算符, 即条件运算符.
- 逗号运算符的优先级最低.
- 逻辑非
!
> 算术运算符 > 关系运算符 > 逻辑与&&
和逻辑或||
> 赋值运算符
运算符的优先级为表达式中的求值顺序提供了重要的依据, 但没有规定所有的顺序.
a = 6 * 5 + 4 * 3;
C 语言没有规定先算 6 * 5 还是先算 4 * 3, 语言的实现者可以根据具体的硬件选择效率最高的计算方式.
再例如:
while (num < 21)
{
printf("%10d %10d\n", num, num * num++);
}
在执行 printf()
语句时, 因为 num
和 num * num++
都是表达式, 会先对两个表达式求值, 再将表达式的值替换格式字符串中的转换说明. 但是具体是先求 num
的值还是先求 num * num++
的值并没有统一的规定, 不同操作系统或不同编译器或不同硬件可能先后顺序不同.
再例如:
n = 3;
y = n++ + n++;
对 y 求值时, 编译器可能使用两次 3, 然后将 n 递增两次. 也可能先使用 n 的原值, 再递增一次 n, 再使用 n 的新值, 再递增一次 n.
虽然最后都会递增两次 n, 但是 y 的值会不一样. 此时的结果是未定义的, 即 C 标准没有规定结果一定是什么.