一.算术操作符
1. 除了 % 操作符之外,其他的几个操作符可以作用于整数和浮点数。
2. 对于 / 操作符如果两个操作数都为整数,执行整数除法。而只要有浮点数执行的就是浮点数除
法。
3. % 操作符的两个操作数必须为整数。返回的是整除之后的余数。
二.移位操作符
<<操作符:左边抛弃、右边补0
要十分注意的一点:整数在内存中存储的是二进制的补码,而printf打印的是原码的值
内存中存放的补码是用来进行数据操作处理的,而打印出来必须是原码用来展示给用户。
而右移操作符分为两种:
1. 逻辑移位
左边用0填充,右边丢弃
2. 算术移位
左边用原该值的符号位填充,右边丢弃
警告⚠ :
对于移位运算符,不要移动负数位,这个是标准未定义的。
四:位操作符
这样的交换方法有一点的弊端,因为整型变量有空间限制,两数相加有可能造成溢出的问题,所以并不是最好的解决方案。
用异或的形式编写才是最佳选择:
可以看出:异或是支持交换律的。
但这样的代码可读性很差,并且只适用于整型变量,实践中效率也不高,所以很少使用。
五.赋值操作符
赋值操作符是一个很棒的操作符,他可以让你得到一个你之前不满意的值。也就是你可以给自己重新赋值。
复合赋值符
+=
-=
*=
/=
%=
>>=
<<=
&=
|=
^=
六.单目操作符
只有一个操作数的操作符
下面对&操作符进行说明:
左值描述的是空间,右值一般指的是空间里的内容
sizeof是操作符,不是函数
sizeof是计算变量或者创建变量的内存大小,和内存中存放什么数据没有关系
~操作符
++操作符:
--操作符跟++操作符使用方式一样
()强制转换操作符
区分逻辑与和按位与
区分逻辑或和按位或
1&2----->0
1&&2---->1
1|2----->3
1||2---->1
下面我们来研究一道面试题:
总结:对于&&操作符,左边一旦遇到假,后面的数就显得不重要了,答案终究是假,就不再往后计算了;||操作符左边一旦遇到真,答案终究是真,就不再往后运算了。
下面我们来看一下逗号操作符:
下标引用、函数调用和结构成员
1.[ ]下标引用
用以上列子很好的说明了,[]只是一个操作符
2.( )函数调用
接受一个或者多个操作数:第一个操作数是函数名,剩余的操作数就是传递给函数的参数。
3.访问一个结构的成员
. 结构体.成员名
-> 结构体指针->成员名
如果我们定义过后想要修改其中的某个数据时应该怎么处理呢?
如果我们想要修改价格,可以如下操作:
但是如果我们想要更改书名和书号像以上这样修改就不可以,因为:b.name和b.id的时候,name和id是一个数组名,它们只是一个空间的起始地址(即首元素地址),我们不能够把想要的东西放入地址中,而是想把它放入这个地址的空间里面去。
这里我们需要用到一个库函数,叫作strcpy(字符串拷贝)
通过以上代码的拷贝我们就很好地将初始化里面的内容进行了修改。
下面我们再来研究一个问题,如果我们给函数传参的时候传的是地址会是什么效果?
以上print2函数中的两种写法访问结构体的内部成员时都是可行的。
由以上可知:结构成员访问操作符
//结构变量.成员名
//结构体指针->成员名
//(*结构体指针).成员名
隐式类型转换
C的整型算术运算总是至少以缺省整型类型的精度来进行的。
为了获得这个精度,表达式中的字符和短整型操作数在使用之前被转换为普通整型,这种转换称为整型提升。
首先来看一下整型提升的意义:
表达式的整型运算要在CPU的相应运算器件内执行,CPU内整型运算器(ALU)的操作数的字节长度
一般就是int的字节长度,同时也是CPU的通用寄存器的长度。
因此,即使两个char类型的相加,在CPU执行时实际上也要先转换为CPU内整型操作数的标准长
度。
通用CPU(general-purpose CPU)是难以直接实现两个8比特字节直接相加运算(虽然机器指令
中可能有这种字节相加指令)。所以,表达式中各种长度可能小于int长度的整型值,都必须先转
换为int或unsigned int,然后才能送入CPU去执行运算。
这样看文字描述可能不好理解,我们来看一段代码深入体会一下这里面的计算过程:
来看看如何进行整型提升:
我们将其分为三种类型:
下面我们来看两个整型提升的实例:
实例1中的a,b要进行整形提升,但是c不需要整形提升
a,b整形提升之后,变成了负数,所以表达式 a==0xb6 , b==0xb600 的结果是假,但是c不发生整形提升,则表达式 c==0xb6000000 的结果是真. 故打印c
实例2中的,c只要参与表达式运算,就会发生整形提升,表达式 +c ,就会发生提升,所以 sizeof(+c) 是4个字节. 表达式 -c 也会发生整形提升,所以 sizeof(-c) 是4个字节,但是 sizeof(c) ,就是1个字节.
算术转换
如果某个操作符的各个操作数属于不同的类型,那么除非其中一个操作数的转换为另一个操作数的类型,否则操作就无法进行。下面的层次体系称为寻常算术转换。
如果某个操作数的类型在上面这个列表中排名较低,那么首先要转换为另外一个操作数的类型后执行运算。
警告:
但是算术转换要合理,要不然会有一些潜在的问题。
操作符的属性
复杂表达式的求值有三个影响的因素。
1. 操作符的优先级
2. 操作符的结合性
3. 是否控制求值顺序
两个相邻的操作符先执行哪个?取决于他们的优先级。如果两者的优先级相同,取决于他们的结合性。
大家可以参考优先级的表: