C语言运算符优先级理解记忆,语法“陷阱”

    运算符优先级最重要两点:

    1.任何一个逻辑运算符的优先级低于任何一个关系运算符。

    2.移位运算符的优先级比算术运算符要低,但是比关系运算符要高。

    记住这两条,能避免很多问题了。

贴上运算符优先级表:

C语言运算符优先级表

优先级

运算符

名称或含义

使用形式

结合方向

说明

1 [] 数组下标 数组名[常量表达式] 左到右 --
() 圆括号 (表达式)/函数名(形参表) --
. 成员选择(对象) 对象.成员名 --
-> 成员选择(指针) 对象指针->成员名 --
 
2 - 负号运算符 -表达式 右到左 单目运算符
~ 按位取反运算符 ~表达式
++ 自增运算符 ++变量名/变量名++
-- 自减运算符 --变量名/变量名--
* 取值运算符 *指针变量
& 取地址运算符 &变量名
! 逻辑非运算符 !表达式
(类型) 强制类型转换 (数据类型)表达式
sizeof 长度运算符 sizeof(表达式)
 
3 / 表达式/表达式 左到右 算数运算符
(双目运算符)
* 表达式*表达式
% 余数(取模) 整型表达式%整型表达式
4 + 表达式+表达式 左到右
- 表达式-表达式
5 <<  左移 变量<<表达式 左到右 移位运算符
(双目运算符)
>>   右移 变量>>表达式
 
6 大于 表达式>表达式 左到右 关系运算符
(双目运算符)
>= 大于等于 表达式>=表达式
小于 表达式<表达式
<= 小于等于 表达式<=表达式
7 == 等于 表达式==表达式 左到右
!= 不等于 表达式!= 表达式
 
8 & 按位与 表达式&表达式 左到右 逻辑运算符
(双目运算符)
9 ^ 按位异或 表达式^表达式 左到右
10 | 按位或 表达式|表达式 左到右
11 && 逻辑与 表达式&&表达式 左到右
12 || 逻辑或 表达式||表达式 左到右
 
13 ?: 条件运算符 表达式1?表达式2:表达式3 右到左 条件运算符
(三目运算符)
 
14 = 赋值运算符 变量=表达式 右到左 赋值运算符
 
15 逗号运算符 表达式,表达式,… 左到右 --

   

下面讲一些理解记忆方式:

A:不同类型的运算符优先级:

1.优先级最高者,其实不是真正意义的运算符,包括:数组下标、函数调用操作符、各结构成员选择操作符。

    它们都是自左于右结合,因此a.b.c的含义是(a.b).c,而不是a.(b.c)

2.在所有的真正意义上的运算符中,单目运算符的优先级最高。

    因为函数调用的优先级要高于单目运算符的优先级,所以如果p是一个函数指针,要调用p所指向的函数,必须这样写:(*p)()。如果写成*p(),编译器会解释成*(p())。

    类型转换符也是单目运算符,它的优先级和其他单目运算符的优先级一样。单目运算符是自右至左,因此*p++会被编译器解释成*(p++),即取指针p所指向的对象,然后将p递增1;而不是(*p)++,即取指针p所指向的对象,然后将该对象递增1。

3.所有双目运算符比单目运算符优先级低。双目运算符中优先级顺序:算数运算符 > 移位运算符 > 关系运算符 > 逻辑运算符 > 赋值运算符。

4.三目运算符 ?:的优先级在所有真正意义的运算符中最低。

    三目运算符优先级最低,所以允许我们在三目条件运算符的条件表达式中包含关系运算符的逻辑组合,例如:

    temp_a = temp_b > 4000 && temp_c < 3000 ?  1:0

    本例也说明,赋值运算符的优先级低于条件运算符的优先级是有意义的。

    此外所有的赋值运算符的优先级是一样的,而它们的结合方式是从右到左。

B:同一类型的运算符间的优先级:

1.算数运算符很好理解,乘法、除法和求余优先级相同。加法、减法优先级相同。两个移位运算符优先级也相同。

2.六个关系运算符的优先级不同。==和!=的优先级要低于其他关系运算符的优先级。

    如果我们要比较a与b的相对大小顺序是否和c与d的相对大小顺序一样,就可以这么写:a < b == c < d

3.任何两个逻辑运算符都具有不同的优先级。所有按位运算符优先级比顺序运算符的优先级高,每个“与”运算符比相应的“或”运算符优先级高,而按位异或(^运算符)的优先级介于按位与运算符和按位或运算符之间。

C:再随便多说两句

用添加括号的方法虽然可以完全避免优先级引发的问题,但是表达式中有了太多的括号反而不容易理解。

涉及到赋值运算符时,经常会引起优先级的混淆,例如下面这个例子

while(c = getc(in) != EOF)

    put(c,out);

在while语句表达式中,c似乎是首先被赋予函数getc(in)的返回值,然后与EOF比较是否达到文件结尾以便决定是否终止循环。然而,由于赋值运算符的优先级要低于任何一个比较运算符,因此c的值实际上是函数getc(in)的返回值与EOF比较的结果。此处函数getc(in)的返回值只是一个临时变量,在与EOF比较后就被“丢弃“了。因此,最后得到的文“副本”中只包括了一组二进制为1的字节流。

你可能感兴趣的:(运算符,优先级,C语言)