操作符基础
- 1. 算数操作符
- 2. 移位操作符
- 3. 位操作符
- 4. 赋值操作符、复合赋值符
- 5. 单目操作符
- 6. 关系操作符
- 7. 逻辑操作符
- 8. 条件操作符
- 9. 逗号表达式
- 10. 下标引用、函数调用、结果成员
- 11. 隐式类型转换
1. 算数操作符
+ - * / %
加减乘除就不用 多说了,这里要说的是 “%”;
%的意思就是取模,也就是取余数;例如:
x=10%3;x的值就是10/3的余数,1
需要注意的点:
a:除了 % 操作符之外,其他的几个操作符可以作用于整数和浮点数。
b:对于 / 操作符如果两个操作数都为整数,执行整数除法。而只要有浮点数执行的就是浮点数除法。
c:% 操作符的两个操作数必须为整数。返回的是整除之后的余数。
2. 移位操作符
<< 左移操作符;左边抛弃,右边补0;(注意,该操作只是操作,不是赋值)
10 << 1 :
10的二进制为 0000 0000 0000 0000 0000 0000 0000 1010
左移一位结果: 0000 0000 0000 0000 0000 0000 0001 0100 ->20(a<< 1,相当于 a*2,但该操作并未有赋值过程,所以a依旧是10,a<<1 的结果才是20)
>>右移操作符;
右移运算分两种:
a:逻辑移位 左边用0填充,右边丢弃(一般为无符号数)
b:算术移位 左边用原该值的符号位填充,右边丢弃(一般为有符号数)
int num=-1;
假设num是-1;这样-1 的补码为全1即:1111 1111 1111 1111 1111 1111 1111 1111;
算术右移结果:1111 1111 1111 1111 1111 1111 1111 1111(左边用原值的符号位填充,右边抛弃)
逻辑右移结果: 0111 1111 1111 1111 1111 1111 1111 1111(坐标补0,右边抛弃)
注意:对于移位运算符,不要移动负数位,这个是标准未定义的。
例如: int num =10;
num >> -1; (这个为错误示范)
3. 位操作符
& 按位与(按比特位,同时为1结果为1,否则结果为0)
例: 15 & 10
15的二进制:1111;
10的二进制:1010;
其结果为:1010 -> 10;
| 按位或(按比特位,同时为0结果为0,否则结果为1)
例: 15 & 10
15的二进制:1111;
10的二进制:1010;
其结果为:1111 -> 15;
^ 按位异或(对应比特位按位异或,相同为0;相异为1) a^0结果为a
例: 15 & 10
15的二进制:1111;
10的二进制:1010;
其结果为:0101 -> 5;
注:他们的操作数必须是整数。
用异或还能不创建临时变量实现两个数的交换
#include <stdio.h>
int main()
{
int a = 10;
int b = 20;
a = a^b;
b = a^b;
a = a^b;
printf("a = %d b = %d\n", a, b);
return 0;
}
4. 赋值操作符、复合赋值符
赋值操作符也都无特殊范例,需要注意的是赋值运算的顺序是从右向左的,而上述的计算操作符以及后续的操作符都是从左向右顺序进行的。
赋值操作符支持连续赋值:
int a=b=c=d=10;(支持,但通常不这样使用)
复合赋值符
+= 例:a+=1 相当于 a=a+1;(后续复合赋值符以此类推)
-=;/=;*=;%=;>>=;<<=;&=;|=;^=;
5. 单目操作符
单目操作符
符号 | 意义 |
! | 逻辑反操作 |
- | 负值 |
+ | 正值 |
& | 取地址 |
sizeof | 求操作数的类型长度(以字节为单位) |
~ | 对一个数按位取反 |
++ | 前置自增或者后置自增 |
- - | 前置自减或者后置自减 |
* | 解引用、间接访问操作符 |
() | 强制类型转换 |
a:前置自增(自减)和后置自增(自减)的区别:
前置自增(自减)为先自增(自减)在使用,后置自增(自减)为先使用,再自增(自减);
#include<stdio.h>
int main()
{
int a=10;
int x1=++a; //先自增,++a为11,再赋给x1;此时x1=11;a=11;
int x2=a++;//先赋值,所以x2=11;然后a自增,所以a输出为12;
printf("x1=%d\n x2=%d\n a=%d",x1,x2,a);
其输出结果为x1=11;x2=11;a=12;
b:强制类型转换(强转和转化的区别:前者本质改动的是类型、后者要改动底层数据;类型其实计算机就是看待数据的方式)
double a=3.14;
int i = (int)d;//将double类型强制转换为int类型;
6. 关系操作符
> ; >= ; < ; <= ; != (用于测试不等) ;==(用于测试相等);
注意:再使用时要防止双等打成单等;
7. 逻辑操作符
a:&&
逻辑与(左右连接的表达式全部为真,则结果为真,否则为假,可连接多个: … && …&& … && …;
b: ||
逻辑或(左右连接的表达式只要有一个为真,则结果为真,否则结果为假,也可连接多个:… || … || … || …;
8. 条件操作符
exp1?exp2:exp3;(exp1如果为真则输出exp2否则输出exp3)
例如:输出两个数的较大数
#include<stdio.h>
int main()
{
int x=0;
int y=0;
printf("比较两个数较大的数,请输入两个操作数# ");
scanf("%d %d",&x,&y);
int num=(x>y?x:y);//x大于y吗?大于输出x;小于输出y;
printf("两个数中较大的是:%d\n",num);
system("pause");
return 0;
}
9. 逗号表达式
exp1,exp2,exp3,exp4, …expN;
逗号表达式的结果以最后一个表达式结果为准;
10. 下标引用、函数调用、结果成员
a:下标引用:[]
数组的引用:数组名+索引值 (arr[2]);
b:函数调用:()
接受一个或者多个操作数:第一个操作数是函数名,剩余的操作数就是传递给函数的参数。( “Game(n)”);
c:结果成员:结构体的访问
. 结构体.成员名
-> 结构体指针->成员名
如:
#include <stdio.h>
struct Stu
{
char name[10];
int age;
}
int main()
{
stu.name = 20;//结构成员访问
pStu->age = 20;//结构成员访问
return 0; //(仅供参考理解访问结构体的方式)
}
11. 隐式类型转换
a:整型提升
C的整型算术运算总是至少以缺省整型类型的精度来进行的。为了获得这个精度,表达式中的字符和短整型操作数在使用之前被转换为普通整型,这种转换称为整型提升。
b:如何进行整型提升
整形提升是按照变量的数据类型(自身)的符号位来提升的(无符号数自动补0)
char a = 0xb6;整型提升为0xFFFF FFB6
short b =0xb600 ;整型提升为0xFFFF B600
整型提升一般考虑的都是变量;整型提升可能会改变数的大小;
char c = 1;
printf("%u\n", sizeof(c));
printf("%u\n", sizeof(+c));//算数运算一定发生整型提升
printf("%u\n", sizeof(!c));//有可能发生整型提升,和平台有关系
c:算数转换
如果某个操作符的各个操作数属于不同的类型,那么除非其中一个操作数的转换为另一个操作数的类型,否则操作就无法进行。下面的层次体系称为寻常算术转换。
long double
double
float
unsigned long int
long int
unsigned int
int
如果某个操作数的类型在上面这个列表中排名较低,那么首先要转换为另外一个操作数的类型后执行运算。
警告: 但是算术转换要合理,要不然会有一些潜在的问题。
d:复杂表达式的求值
有三个影响的因素: 操作符的优先级、操作符的结合性、是否控制求值顺序。
e:两个相邻的操作符先执行哪个?
取决于他们的优先级。如果两者的优先级相同,取决于他们的结合性。
注:各位读者阅读完本文,如果对你有所帮助,还请一键三连哦!如果文章有不妥之处,欢迎在下方评论指出!