1. 操作符的分类
2. ⼆进制和进制转换
3. 原码、反码、补码
4. 移位操作符
5. 位操作符:&、|、^、~
6. 单⽬操作符
7. 逗号表达式
8. 下标访问[]、函数调⽤()
9. 结构成员访问操作符
10. 操作符的属性:优先级、结合性
11. 表达式求值
1. 操作符的分类
2. ⼆进制和进制转换
为了区分不同进制
规定8进制需要在前面加上0
16进制需要加上0x
15 的 2 进制: 1111 满2进1位 0 1
15 的 8 进制: 017 满8进1位 0 1 2 3 4 5 6 7
15 的 10 进制: 15 满10进1位 0 1 2 3 4 5 6 7 8 9
15 的 16 进制:0xF 满16进1位 01 2 3 4 5 6 7 8 9 a b c d e f 或(0 1 2 3 4 5 6 7 8 9 A B C D E F)
2.1 2进制转10进制
举例 列如
0 0 0 0 1 0 1 0
二进制位从最右边开始数是2的0次方,2的1次方依次递增 有1则计算 为0则不计算
如上方 是第二位和第四位是1 则计算 2的1次方+2的3次方= 2+8=10
转换位10进制就是 10。
2.1.1 10进制转2进制数字
2.2 2进制转8进制和16进制
2.2.1 2进制转8进制
8进制的数字每⼀位是0~7的,0~7的数字,各⾃写成2进制,最多有3个2进制位就⾜够了
2.2.2 2进制转16进制
16进制的数字每⼀位是0~9,a~f的,0~9,a~f的数字,各⾃写成2进制,最多有4个2进制位就⾜够了
3. 原码、反码、补码
正整数的原码补码反码都一样
负整数的原码补码反码参考一以下
原码:直接将数值按照正负数的形式翻译成⼆进制得到的就是原码。
反码:将原码的符号位不变,其他位依次按位取反就可以得到反码。
补码:反码+1就得到补码
补码得到原码也是可以使⽤:+1,取反的操作。
原码 00000000 00000011 11101011 11011111
反码 11111111 11111100 00010100 00100000
补码 11111111 11111100 00010100 00100001
4. 移位操作符
<<左移操作符
>> 右移操作符
注: 移位操作符的操作数只能是整数。
移动的是二进制的数
4.1 左移操作符
移位规则:左边抛弃、右边补0
4.2 右移操作符
移位规则:⾸先右移运算分两种:
1. 逻辑右移:左边⽤0填充,右边丢弃 左移操作符演⽰
2. 算术右移:左边⽤原该值的符号位填充,右边丢弃
5. 位操作符:&、|、^、~
& 按位与 有0出0
| 按位或 有1 出1
^ 按位异或 相同出0 不同出1
~ 按位取反 1变0 0变1
注: 他们的操作数必须是整数。
案例1
不能创建临时变量(第三个变量),实现两个数的交换。
但可能存在溢出的方法
使用按位异或能完美的解决
案例2
编写代码将13⼆进制序列的第5位修改为1,然后再改回0
13 的 2 进制序列: 00000000000000000000000000001101
将第 5 位置为 1 后: 00000000000000000000000000011101
将第 5 位再置为 0 : 00000000000000000000000000001101
6. 单⽬操作符
!、 ++ 、- 、 & 、 * 、 + 、 、 ~ 、 sizeof 、 ( 类型 )
7. 逗号表达式
、
逗号表达式,就是⽤逗号隔开的多个表达式,整个表达式的结果是最后⼀个表达式的结果。
举例:
8. 下标访问[]、函数调⽤()
8.1[ ] 下标引⽤操作符
操作数:⼀个数组名+⼀个索引值
8.2 函数调⽤操作符
接受⼀个或者多个操作数:第⼀个操作数是函数名,剩余的操作数就是传递给函数的参数
9. 结构成员访问操作符
9.1 结构体
9.1.1 结构的声明
注意最后的分号不能丢。
9.1.2 结构体变量的定义和初始化
定义
初始化
9.2 结构成员访问操作符
9.2.1直接访问
结构体成员的直接访问是通过点操作符(.)访问的。点操作符接受两个操作数
使⽤⽅式:结构体变量.成员名
9.2.2 结构体成员的间接访问
使⽤⽅式:结构体指针->成员名
10. 操作符的属性:优先级、结合性
10.1 优先级
例如 3+4*5
由于乘法 的优先级⾼于加法,所以会先计算4*5在计算3+4
10.2 结合性
如果两个运算符优先级相同,优先级没办法确定先计算哪个了,这时候就看结合性了,则根据运算符 是左结合,还是右结合,决定执⾏顺序。⼤部分运算符是左结合(从左到右执⾏),少数运算符是右 结合(从右到左执⾏),⽐如赋值运算符( = )
11. 表达式求值
11.1 整型提升
表达式中各种⻓度可能⼩于int⻓度的整型值,都必须先转换为 int或unsigned int,然后才能送⼊CPU去执⾏运算
如何进⾏整体提升呢?
1.有符号整数提升是按照变量的数据类型的符号位来提升的
2. ⽆符号整数提升,⾼位补0
11.2 算术转换
如果某个操作符的各个操作数属于不同的类型,那么除⾮其中⼀个操作数的转换为另⼀个操作数的类 型,否则操作就⽆法进⾏。下⾯的层次体系称为寻常算术转换
按照上图从上往下转换
11.3 问题表达式解析
11.3.1 表达式1
*比+优先级高所以先执行*,但是优先级并不 能决定第三个*⽐第⼀个+早执⾏
所以执行的顺序可能变成
a*b
c*d
a*b + c*d
e*f
a*b + c*d + e*f
11.3.2 表达式2
--的优先级高于+,但是我们并没有办法得知, +操作符的左操作数的获取在右操作数之前还是之后求值,所以结果是不可预测的
11.3.3 表达式3
表达式3在不同编译器中测试结果:⾮法表达式程序的结果
11.3.4 表达式4
answer = fun() - fun() * fun(); 中我们只能通过操作符的优先级得知:先 算乘法,再算减法。 函数的调⽤先后顺序⽆法通过操作符的优先级确定
11.3.5 表达式5:
这段代码中的第⼀个 + 在执⾏的时候,第三个++是否执⾏,这个是不确定的,因为依靠操作符的优先 级和结合性是⽆法决定第⼀个 + 和第三个前置 ++ 的先后顺序