第14课 - 优先级和类型转换分析
问题引入:
#include <stdio.h>
#include <malloc.h>
typedef struct _demo
{
int* pInt;
float f;
} Demo;
int func(int v, int m)
{
return ((v & m )!= 0);
}
int main()
{
Demo* pD = (Demo*)malloc(sizeof(Demo));
int *p[5]; //int* p[5];
int *f(); //int* f()
int i = 0;
i = 1, 2; // (i = 1), 2;
// *pD.f = 0;
pD->f = 0;
free(pD);
return 0;
}
- 优先级别表格
优先级 |
运算符 |
名称或含义 |
使用形式 |
结合方向 |
说明 |
1 |
[] |
数组下标 |
数组名[常量表达式] |
左到右 |
|
() |
圆括号 |
(表达式)/函数名(形参表) |
|
||
. |
成员选择(对象) |
对象.成员名 |
|
||
-> |
成员选择(指针) |
对象指针->成员名 |
|
||
++ |
后置自增运算符 |
++变量名 |
单目运算符 |
||
-- |
后置自减运算符 |
--变量名 |
单目运算符 |
||
2 |
- |
负号运算符 |
-表达式 |
右到左 |
单目运算符 |
(类型) |
强制类型转换 |
(数据类型)表达式 |
|
||
++ |
前置自增运算符 |
变量名++ |
单目运算符 |
||
-- |
前置自减运算符 |
变量名-- |
单目运算符 |
||
* |
取值运算符 |
*指针变量 |
单目运算符 |
||
& |
取地址运算符 |
&变量名 |
单目运算符 |
||
! |
逻辑非运算符 |
!表达式 |
单目运算符 |
||
~ |
按位取反运算符 |
~表达式 |
单目运算符 |
||
sizeof |
长度运算符 |
sizeof(表达式) |
|
||
3 |
/ |
除 |
表达式/表达式 |
左到右 |
双目运算符 |
* |
乘 |
表达式*表达式 |
双目运算符 |
||
% |
余数(取模) |
整型表达式/整型表达式 |
双目运算符 |
||
4 |
+ |
加 |
表达式+表达式 |
左到右 |
双目运算符 |
- |
减 |
表达式-表达式 |
双目运算符 |
||
5 |
<< |
左移 |
变量<<表达式 |
左到右 |
双目运算符 |
>> |
右移 |
变量>>表达式 |
双目运算符 |
||
6 |
> |
大于 |
表达式>表达式 |
左到右 |
双目运算符 |
>= |
大于等于 |
表达式>=表达式 |
双目运算符 |
||
< |
小于 |
表达式<表达式 |
双目运算符 |
||
<= |
小于等于 |
表达式<=表达式 |
双目运算符 |
||
7 |
== |
等于 |
表达式==表达式 |
左到右 |
双目运算符 |
!= |
不等于 |
表达式!= 表达式 |
双目运算符 |
||
8 |
& |
按位与 |
表达式&表达式 |
左到右 |
双目运算符 |
9 |
^ |
按位异或 |
表达式^表达式 |
左到右 |
双目运算符 |
10 |
| |
按位或 |
表达式|表达式 |
左到右 |
双目运算符 |
11 |
&& |
逻辑与 |
表达式&&表达式 |
左到右 |
双目运算符 |
12 |
|| |
逻辑或 |
表达式||表达式 |
左到右 |
双目运算符 |
13 |
?: |
条件运算符 |
表达式1? 表达式2: 表达式3 |
右到左 |
三目运算符 |
14 |
= |
赋值运算符 |
变量=表达式 |
右到左 |
|
/= |
除后赋值 |
变量/=表达式 |
|
||
*= |
乘后赋值 |
变量*=表达式 |
|
||
%= |
取模后赋值 |
变量%=表达式 |
|
||
+= |
加后赋值 |
变量+=表达式 |
|
||
-= |
减后赋值 |
变量-=表达式 |
|
||
<<= |
左移后赋值 |
变量<<=表达式 |
|
||
>>= |
右移后赋值 |
变量>>=表达式 |
|
||
&= |
按位与后赋值 |
变量&=表达式 |
|
||
^= |
按位异或后赋值 |
变量^=表达式 |
|
||
|= |
按位或后赋值 |
变量|=表达式 |
|
||
15 |
, |
逗号运算符 |
表达式,表达式,… |
左到右 |
从左向右顺序运算 |
- 易错的优先级
优先级问题 |
表达式 |
经常误认为的结果 |
实际结果 |
.的优先级高于* ->操作符用于消除这个问题 |
*p.f |
P所指向的字段f (*P).p |
对p取f偏移,作为指针,然后进行解除引用操作。*(p.f) |
[]高于* |
int *ap[] |
ap是个指向int数组的指针 int (*fp)() |
ap是个元素为int指针的数组 int *(ap[]) |
函数()高于* |
int *fp() |
fp是个函数指针,所指的函数返回int。 int (*fp)() |
fp是个函数,返回值是int * int *(fp()) |
==和!=高于操作位 |
(val & mask != 0) |
(val & mask) != 0 |
val & (mask != 0) |
==和!=高于赋值符 |
c = getchar() !=EOF |
(c = getchar()) !=EOF |
c = (getchar() !=EOF) |
算数运算符高于位移运算符 |
msb << 4 + 1sb |
(msb << 4) + 1sb |
msb << (4 + 1sb) |
逗号运算符在所有运算符中有心级别最低 |
i = 1,2 |
i = ( 1,2) |
(i = 1), 2 |
- C语言中隐式的类型转换
(1) 算术运算式中,低类型转换为高类型。
(2) 赋值表达式中,表达式的值转换为左边变量的类型。
(3) 函数调用时,实参转换为形参的类型。
(4) 函数返回值,return表达式转换为返回值类型。
char + short = int
- 实例分析
隐式类型转换带来困惑
#include <stdio.h>
int main()
{
int i = -2;
unsigned int j = 1;
if( (i + j) >= 0 )
{
printf("i+j>=0 ");
}
else
{
printf("i+j<0 ");
}
printf("i+j=%d ", i + j);
return 0;
}
运行结果:i+j>=0
i+j=-1
-2和1相加的时候,-2被转化为unsigned int,-2的补码是30个1和10,共32位。也就是0xFFFFFFFE,把它看成是无符号整数就会很大,所以会打印i+j>=0。int型-1的补码是0xFFFFFFFF,但是%d表示以int型打印,所以会打印出来-1。%s代表unsigned int。
改成printf("i+j=%0x ", i + j);运行结果是:i+j=0xFFFFFFFF。