一、隐式类型转换,强制类型转换
从左往右,必转,数据越精确优先级越高。
不同精度数据类型进行混合运算,就会发生隐式转换。
典型例子
1.整型提升
short a = 10;
int b = 5;
int c = a + b; // a会被隐式提升为int类型,与b相加得到结果
2.浮点数提升
float a = 3.14;
double b = 2.5;
double c = a + b; // a会被隐式提升为double类型,与b相加得到结果
这是很反直觉的,所以隐式转换这里特别容易出问题 。
这个方法可以输出123.45 而不会输出123.46 因为计算机会默认四舍五入。
二、左值右值
lvalue(左值):lvalue 是指在赋值操作中出现在等号左侧的表达式,它通常代表一个可被赋值的内存位置。换句话说,lvalue 是一个可以出现在赋值号的左边的表达式,它通常具有持久性,可以被取地址。但如果加了const ,那么就会移除左值的性质 不能赋值 但可以取地址。
int x = 5; // x 是一个左值,可以进行赋值操作
rvalue(右值):rvalue 是指在赋值操作中出现在等号右侧的表达式,它通常代表一个临时的计算结果或者一个常量。rvalue 表示一个临时的值,而不是一个可修改的内存位置。
int y = 10 + 5; // 10 + 5 是一个右值,表示一个临时的计算结果
表达式和匿名变量都是右值,不能被& 取地址。
三、%取余的使用
%用于计算两个整数相除后的余数。
- 不能用于浮点型
- 取余运算与/ (除法)的第二个操作数都不能为0
- 取余运算 的运算结果的符号 取决于左操作数的符号。
例如-10%3 ——》-1
10%-3 ——》1
-10%-3——》-1
%的常用用法
- 用来判断奇数偶数
int num = 7; if (num % 2 == 0) { printf("%d 是偶数\n", num); } else { printf("%d 是奇数\n", num); }
-
a%N 结果 被限制在 [0~N-1] 范围内
随机输出10个 被限制在50~100的整数。
*******************************************************************
补充:
i+++++i
运算符服从贪心原则
从左到右逐个字符的扫描表达式,看这些字符能否,
尽可能多的结合成c的运算符 不能结合,就还是字符本身
分解为
i ++ ++ + i
((i++)++)+i
i++完成运算后 会变成右值,但++要求必须与左值(变量)相结合,所以会报错。
应避免 printf("%d%d\n",i,++i); 这种代码的出现
因为格式字符串中的两个%d
格式指示符对应的参数依赖于它们在参数列表中的位置。然而,参数的求值顺序在C语言中是未定义的。这意味着编译器可以选择任意的求值顺序,这包括先求值i
还是先求值++i
。这种未定义的行为会导致程序的行为不确定,可能会产生错误的结果。
使用运算符时 要格外注意它的优先级以及结合顺序。