操作符详解
操作符和表达式
操作符
算术操作符
移位操作符
位操作符
赋值操作符
单目操作符
关系操作符
逻辑操作符
条件操作符
表达式
逗号表达式
下标引用、函数调用和结构成员
一.算术操作符
- 除了 % 操作符之外,其他的几个操作符可以作用于整数和浮点数。
- 对于 / 操作符如果两个操作数都为整数,执行整数除法。而只要有浮点数执行的就是浮点数除法。
- % 操作符的两个操作数必须为整数。返回的是整除之后的余数。
二.移位操作符
1.<< 左移操作符
2.>>右移操作符
左移操作符 移位规则:左边抛弃、右边补0
右移操作符移位规则:首先右移运算分两种:
- 逻辑移位 左边用0填充,右边丢弃
- 算术移位 左边用原该值的符号位填充,右边丢弃
三.位操作符
& :按位与
| :按位或
^ :按位异或
注:他们的操作数必须是整数。
10&20; 10|20; 10^20:
与位运算的相关面试题:点击获取
补充:不能创建临时变量(第三个变量),实现两个数的交换:
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;
}
四.赋值操作符
复合赋值符:
+=,-=,*=,/=,%=,>>=,<<=,&=,|=,^=
int x = 10;
x = x+10;
x += 10;复合赋值
其他运算符一样的道理。这样写更加简洁。
五.单目操作符
补充:
- sizeof操作符:
int arr[10] = { 0 };
printf("%d", sizeof(arr[10]));
一般思维认为10越界,就会出现未定义行为,但是这里不是,只因为sizeof操作符是在编译期执行,而数组下标越界是在执行期间产生,这时sizeof已经执行完了,所以根本不会越界
- 将前置++,后置++写成函数就是下面两个函数
++函数(num){
num+= 1;
return num;
}
函数++(num){
int temp = num;
num=num+1;
return temp;
}
- 取地址和解引用后面详细解释在其他版块
六.关系操作符
在编程的过程中== 和=不小心写错,导致的错误
七.逻辑操作符
记住短路求值规则
八.条件操作符
这个用法很简单,但是用处有时很大,可以起到压缩代码的作用
九.逗号表达式
1.逗号表达式,就是用逗号隔开的多个表达式。 逗号表达式,从左向右依次执行。整个表达式的结果是最后一个表达式的结果。
2.逗号表达式,在python中比较有用
十.下标引用、函数调用和结构成员
- [ ]下标引用操作符
操作数:一个数组名 + 一个索引值
- ( ) 函数调用操作符 接受一个或者多个操作数:第一个操作数是函数名,剩余的操作数就是传递给函数的参数
- 访问一个结构的成员
. 结构体.成员名
-> 结构体指针->成员名
后面在结构体章节有才详细介绍
十一.表达式求值
表达式求值的顺序一部分是由操作符的优先级和结合性决定
同样,有些表达式的操作数在求值的过程中可能需要转换为其他类型
隐式类型转换:
C的整型算术运算总是至少以缺省整型类型的精度来进行的。为了获得这个精度,表达式中的字符和短整型操作数在使用之前被转换为普通整型,这种转换称为整型提升。
整型提升的意义:
表达式的整型运算要在CPU的相应运算器件内执行,CPU内整型运算器(ALU)的操作数的字节长度一般就是int的字节长度,同时也是CPU的通用寄存器的长度。因此,即使两个char类型的相加,在CPU执行时实际上也要先转换为CPU内整型操作数的标准长度。通用CPU(general-purpose CPU)是难以直接实现两个8比特字节直接相加运算(虽然机器指令中可能有这种字节相加指令)。所以,表达式中各种长度可能小于int长度的整型值,都必须先转换为int或unsigned int,然后才能送入CPU去执行运算.
点击了解整形提升讲解
总结补充操作符的属性
1.复杂表达式的求值有三个影响的因素:
- 操作符的优先级
- 操作符的结合性
- 是否控制求值顺序。
两个相邻的操作符先执行哪个?取决于他们的优先级。如果两者的优先级相同,取决于他们的结合性。
++i+ i++这种就是有问题,结合性与优先级冲突,是未定义行为
2.const修饰的变量是不可以进行第二次赋值的
eg:const int a = 100;
a = 20;//这句话是编译不过的
3.int所占字节
在32位系统中,int占4个字节
在64位系统中,int占4个字节
4.可以利用宏来定义&&和||
#define and &&
#define or ||
很多都可以用宏来简化