系列文章目录
C语言小白急救 操作符详解(8千字保姆级教程)
C语言小白急救 扫雷游戏(万字保姆级教程)
C语言小白急救 使用C语言编写‘三子棋‘
前言
表达式求值的顺序一部分是由操作符的优先级和结合性决定。
此外,有些表达式的操作数在求值的过程中可能需要转换为其他类型。
一、 隐式类型转换
1.整型提升
概念:
C的整型算术运算总是至少以缺省整型类型的精度来进行的,为了获得这个精度,表达式中的字符和短整型操作数在使用之前被转换为普通整型,这种转换称为整型提升。
(即在C中的计算一般是要先转化为整形(int or unsigned int)再进行计算)
表达式的整型运算要在CPU的相应运算器件内执行,CPU内整型运算器(ALU)的操作数的字节长度
一般就是int的字节长度,同时也是CPU的通用寄存器的长度。
因此,即使两个char类型的相加,在CPU执行时实际上也要先转换为CPU内整型操作数的标准长
度。
通用CPU(general-purpose CPU)是难以直接实现两个8比特字节直接相加运算(虽然机器指令
中可能有这种字节相加指令)。所以,表达式中各种长度可能小于int长度的整型值,都必须先转
换为int或unsigned int,然后才能送入CPU去执行运算。
怎么进行整形提升
整形提升是按照变量的数据类型的符号位来提升的(正数为0,负数为1)
例如:
char a = 5; //补码:00000101
char b = 127;//补码:01111111
char c = a + b;
//a提升后补码:000000000000000000000000 00000101
//b提升后补码:000000000000000000000000 01111111
//a + b 补码:000000000000000000000000 10000100
//c 的补码:10000100
//c 的反码:10000011
//c 的原码:11111100
printf("%d",c);
不知道原、反、补码的请移步:C语言小白急救 操作符详解(8千字保姆级教程)
例题1:
//打印出的值为?
int main()
{
char c = 1;
printf("%u\n", sizeof(c));
printf("%u\n", sizeof(+c));
printf("%u\n", sizeof(-c));
return 0;
}
解析:,c只要参与表达式运算,就会发生整形提升,表达式 +c ,就会发生提升,所以 sizeof(+c) 是4个字节;表达式 -c 也会发生整形提升,所以 sizeof(-c) 是4个字节;但是 c没有进行运算,所以sizeof© ,就是1个字节.
二.算术转换
概念:
如果某个操作符的各个操作数属于不同的类型,那么除非其中一个操作数的转换为另一个操作数的类型,否则操作就无法进行,下面的层次体系称为寻常算术转换。
(当已经>=整形时考虑),算数转换的方式与整形提升相同。
long double
double
float
unsigned long int
long int
unsigned int
int
警告:
但是算术转换要合理,要不然会有一些潜在的问题。
float f = 3.14;
int num = f;//隐式转换,会有精度丢失
三、 操作符的属性
复杂表达式的求值有三个影响的因素。
- 操作符的优先级
- 操作符的结合性
- 是否控制求值顺序。
两个相邻的操作符先执行哪个?取决于他们的优先级。如果两者的优先级相同,取决于他们的结合性。
执行过程:
(1)相邻操作符,优先级高的先执行
(2)当优先级相同时,以结合性判断怎么执行
(3)已知优先级与结合性,但是不一定就能确定一个式子的执行顺序
例如:
a*b + c*d + e*f
解析:
在计算的时候,由于比+的优先级高,只能保证,的计算是比+早,但是优先级并不
能决定第三个比第一个+早执行。
计算结果可能是:
b
cd
ab + cd
ef
ab + cd + e*f
或者:
ab
cd
ef
ab + cd
ab + cd + ef