328寝室收集整理
在以后的学习和开发的生活中,大家难免会因为运算符优先级和结合性犯一些小错误。尤其是在学习指针部分后,解引用运算符(取值运算符)和取下标运算符、函数调用运算符、自增自减运算符等之间容易因为优先级等容易混淆的问题而深受困扰。
如:&(*(*(array + i))+ y) 的含义,* str[i] 的含义,(*str)[i] 的含义,*i++的含义
由数组与指针之间像奶妈跟正太之间基友般的关系可知
&(*(*(array + i))+ y) 等价于 &array[i][y]
而当y = 0 时 &(*(*(array + i))+ y)= &(*array [ i]) = array[i]
这就可以说明为什么array[i]也指向第一个元素的地址了。
而* str[i] 实际上是指针数组,用指针来表示就是**str (指针的指针)。这里要注意(*str)[i]和* str[i] 不等同,因为* 与[] 的优先级不同(低)。(*str)[i] 其实是 str[][i] ,也就是多维数组。而在指针里表达的意思是指向内存空间为 i 倍相应类型的长度的地址。例如:
char (*str)[i];
则 str[j] = (*str + j * i) str[j][d] = *str[j] = *(*str + j*i) = *(*(str + j) + d)
注意:数组多维(N)数组名指向的是数组第N维的元素首地址,而不是数组的第一个元素的地址,如str 指向的是 str[0],而不是str[0][0]。造成这种事实的原因是C语言并不认为数组是多维的,而是数组元素又是数组。
*i++ 由运算符的优先级可知,*i++其实为 *(i++) ,也就是先自增运算后解引用。由后置自增运算符的性质知,自增运算的表达式返回值其实是自增之前的指,所以*i++表达式的返回值是i自增之前的地址所指向的值(注意:此时i已经改变)。
以上列举的是一些常见的数组与指针关系的问题,如果遇到函数指针的问题,有时也是相当棘手的。
如:( *(void (*)()) 0 )(); 的含义
其实上述式子是一个语句,而且是一个函数的调用语句,调用的是以0为地址返回值为void的函数指针所指向的函数。
在C语言中,指针很重要,也是一个难点,在很多的开发中都能用到,而且C语言之所以强大,靠的不止是C语言简洁的语法,很大程度是因为C指针的强大。C作为高级语言的存在,而且是最接近硬件的高级语言,C指针在硬件访问中扮有很重要的角色。
优先级 |
运算符 |
名称或含义 |
使用形式 |
结合方向 |
说明 |
1 |
[] |
数组下标 |
数组名[常量表达式] |
左到右 |
|
() |
圆括号 |
(表达式)/函数名(形参表) |
|||
. |
成员选择(对象) |
对象.成员名 |
|||
-> |
成员选择(指针) |
对象指针->成员名 |
|||
2 |
- |
负号运算符 |
-表达式 |
右到左 |
单目运算符 |
(类型) |
强制类型转换 |
(数据类型)表达式 |
|||
++ |
自增运算符 |
++变量名/变量名++ |
单目运算符 |
||
-- |
自减运算符 |
--变量名/变量名-- |
单目运算符 |
||
* |
取值运算符 |
*指针变量 |
单目运算符 |
||
& |
取地址运算符 |
&变量名 |
单目运算符 |
||
! |
逻辑非运算符 |
!表达式 |
单目运算符 |
||
~ |
按位取反运算符 |
~表达式 |
单目运算符 |
||
sizeof |
长度运算符 |
sizeof(表达式) |
|||
3 |
/ |
除 |
表达式/表达式 |
左到右 |
双目运算符 |
* |
乘 |
表达式*表达式 |
双目运算符 |
||
% |
余数(取模) |
整型表达式/整型表达式 |
双目运算符 |
||
4 |
+ |
加 |
表达式+表达式 |
左到右 |
双目运算符 |
- |
减 |
表达式-表达式 |
双目运算符 |
||
5 |
<< |
左移 |
变量<<表达式 |
左到右 |
双目运算符 |
>> |
右移 |
变量>>表达式 |
双目运算符 |
||
6 |
> |
大于 |
表达式>表达式 |
左到右 |
双目运算符 |
>= |
大于等于 |
表达式>=表达式 |
双目运算符 |
||
< |
小于 |
表达式<表达式 |
双目运算符 |
||
<= |
小于等于 |
表达式<=表达式 |
双目运算符 |
||
7 |
== |
等于 |
表达式==表达式 |
左到右 |
双目运算符 |
!= |
不等于 |
表达式!= 表达式 |
双目运算符 |
||
8 |
& |
按位与 |
表达式&表达式 |
左到右 |
双目运算符 |
9 |
^ |
按位异或 |
表达式^表达式 |
左到右 |
双目运算符 |
10 |
| |
按位或 |
表达式|表达式 |
左到右 |
双目运算符 |
11 |
&& |
逻辑与 |
表达式&&表达式 |
左到右 |
双目运算符 |
12 |
|| |
逻辑或 |
表达式||表达式 |
左到右 |
双目运算符 |
13 |
?: |
条件运算符 |
表达式1? 表达式2: 表达式3 |
右到左 |
三目运算符 |
14 |
= |
赋值运算符 |
变量=表达式 |
右到左 |
|
/= |
除后赋值 |
变量/=表达式 |
|||
*= |
乘后赋值 |
变量*=表达式 |
|||
%= |
取模后赋值 |
变量%=表达式 |
|||
+= |
加后赋值 |
变量+=表达式 |
|||
-= |
减后赋值 |
变量-=表达式 |
|||
<<= |
左移后赋值 |
变量<<=表达式 |
|||
>>= |
右移后赋值 |
变量>>=表达式 |
|||
&= |
按位与后赋值 |
变量&=表达式 |
|||
^= |
按位异或后赋值 |
变量^=表达式 |
|||
|= |
按位或后赋值 |
变量|=表达式 |
|||
15 |
, |
逗号运算符 |
表达式,表达式,… |
左到右 |
从左向右顺序运算 |
说明:
同一优先级的运算符,运算次序由结合方向所决定。
简单记就是:! > 算术运算符 > 关系运算符 > && > || > 赋值运算符
参考文献:
K.N.King C Programming: A Modern Approach, Second Edition
Koening,A. C Traps and Pitfalls
谭浩强 C程序设计教程
以上内容由计科124-10- 328寝室 整理
感谢Jakes为本文前述撰文,感谢 Arthas , Jack , Huasheng参与交流,
感谢 Hehe , Guodong 给予的支持,感谢 Room&Long 帅气的海飞丝
2012年12月1日 0:52 距离世界末日还有 ** 天
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于