目录
1.操作符的属性
操作符有两种属性:优先级和结合性。
1.1优先级
优先级指的是,如果⼀个表达式包含多个运算符,哪个运算符应该优先执行。各种运算符的优先级是不⼀样的。
//示例1
int a;
a=1+3*4;//表达式中既有加分运算符(+),也有乘法运算符(*),由于乘法运算符优先级高于加法,所以先算乘法(3*4)
//示例2
int a=1,c;
c=3*a++;//表达式中既有单目操作符(++),也有乘法运算符(*),由于自增后缀优先级高于乘法,所以先算自增后缀(a++)
参考链接:https://zh.cppreference.com/w/c/language/operator_precedence
1.2结合性
如果两个运算符优先级相同,优先级没办法确定先计算哪个了,这时候就看结合性了,则根据运算符是左结合,还是右结合,决定执行顺序。大部分运算符是左结合(从左到右执行),少数运算符是右结合(从右到左执行),比如赋值运算符(=)。
//示例
int a;
a=3*4/5; // * 和 / 的优先级相同,它们都是左结合运算符,所以从左到右执⾏,先算3*4。
注:由于圆括号(())优先级最高,所以可以使用圆括号改变其他运算符的优先级。
2.表达式求值
2.1整型提升
C语言中整型算术运算总是至少以缺省整型类型的精度来进行的。 为了获得这个精度,表达式中的字符和短整型操作数在使用之前被转换为普通整型,这种转换称为整型提升。
提升规则:
1.有符号整数提升是按照变量的数据类型的符号位来提升的
2.无符号整数提升,高位补0
//示例
#include<stdio.h>
int main()
{
//3和127均为整型(int),有32个比特位;a,b,c为字符,只有8个比特位,而输出结果是用的%d,所以这里就用到了整型提升
char a = 3;
//00000000000000000000000000000011 — 3为正数,原反补码相同
//00000011 — a只有8比特位,前24位被截断,取最后8位(此char为sign char,有符号整数提升是按照变量的数据类型的符号位来提升的)
char b = 127;
//00000000000000000000000001111111 — 127为正数,原反补码相同
//01111111 — b的补码(解释同上)
char c = a + b;
//00000011 — a
//01111111 — b
//00000000000000000000000000000011 — a整型提升
//00000000000000000000000001111111 — b整型提升
//00000000000000000000000010000010 — 整型相加
//10000010 — c的补码(解释同上)
//%d — 以十进制打印有符号整数
//11111111111111111111111110000010 — 整数提升(补码)
//10000000000000000000000001111101 — 反码
//10000000000000000000000001111110 — 原码(结果为-126)
printf("%d", c);
return 0;
}
注:char是指sign char还是unsigned char是取决于编译器的实现,但是常见的编译器上char ——> sign char,例如vs2022
2.2算数转换
如果某个操作符的各个操作数属于不同的类型,那么除非其中⼀个操作数的转换为另⼀个操作数的类型,否则操作就无法进行。下面的层次体系称为寻常算术转换。