C语言—操作符详解

一、操作符的分类

• 算术操作符: +(加)、-(减)、*(乘)、/(除)、%(取余)
• 移位操作符:  << (左移操作符)  >>(右移操作符) 
• 位操作符:  &(按位与)  |(按位或)  ^(按位异或) 
• 赋值操作符:  = 、+= 、 -= 、 *= 、 /= 、%= 、<<= 、>>= 、&= 、|= 、^=

• 单⽬操作符: !(取反)、++(自加1,分为前置和后置)、--(自减1,分为前置和后置)、&(取地址)、*(解引用)、+(正)、-(负) 、~(按位取反) 、sizeof、(类型) <-(强制类型转换)
• 关系操作符:  > 、>= 、< 、<= 、 == 、 != 
• 逻辑操作符: && (并且)、||(或者)
• 条件操作符: ? :(三目表达式)
• 逗号表达式: ,
• 下标引⽤: [ ]
• 函数调⽤: ( )

• 结构成员访问:    ->

二、二进制和进制转换

我们常听到的2进制、8进制、10进制、16进制都是数值的不同表⽰形式⽽已。

例如:数值10的各种进制的表⽰形式:

这里我们重点介绍⼀下⼆进制:

类比十进制,十进制有两个特点:

• 10进制中满10进1
• 10进制的数字每⼀位都是0~9的数字组成

其实二进制也是一样的:

• 2进制中满2进1
• 2进制的数字每⼀位都是0~1的数字组成

(2.1)二进制转十进制

首先我们要了解十进制的数字到底是怎么表示的,例如10进制的123表⽰的值是⼀百⼆⼗三,为什么是这个值呢?其实10进制的每⼀位是有权重的,10进制的数字从右向左是个位、⼗位、百位....,分别每⼀位的权重是10 的零次幂,10的一次幂,10的二次幂。如下图:

2进制和10进制是类似的,只不过2进制的每⼀位的权重,从右向左是:2的零次幂,2的一次幂,2的二次幂......

例如:2进制的1101转换为十进制

(2.1.1)十进制转二进制数字

方法如下图:

解释:十进制转换为二进制就是用十进制数字除2取余,再用商再次除2取余,直到商为0为止,这时就像上图一样,将得到的余数由下往上写出,就是十进制转换出的二进制。

(2.2)二进制转八进制和十六进制

(2.2.1)二进制转八进制

8进制的数字每⼀位是0~7的,0~7的数字,各⾃写成2进制,最多有3个2进制位就⾜够了,这是为什么呢?在八进制数字中,7是最大的数字了,而7的⼆进制是111(即2的0次幂*1+2的1次幂*1+2的2次幂*1),所以八进制转二进制最多需要三个二进制位,所以在2进制转8进制数的时候,从2进制序列中右边低位开始向左每3个2进制位会换算⼀个8进制位,剩余不够3个2进制位的直接换算。

如:2进制的01101011,换成8进制:0153,0开头的数字,会被当做8进制。

注:八进制转二进制同理,每一个八进制数字都转换为三个二进制位。

(2.2.2)二进制转十六进制

16进制的数字每⼀位是0~9 , a~f 组成的,0~9,a~f 的数字,各⾃写成2进制,最多有4个2进制位就⾜够了(与八进制同理),⽐如 f(即15) 的⼆进制是1111,所以在2进制转16进制数的时候,从2进制序列中右边低位开始向左每4个2进制位会换算⼀个16进制位,剩余不够4个⼆进制位的直接换算。

如:2进制的01101011,换成16进制:0x6b,16进制表⽰的时候前⾯加0x

注:十六进制转二进制同理,每一个十六进制数字都转换为四个二进制位。

三、源码,反码,补码

整数的2进制表⽰⽅法有三种,即原码、反码和补码
有符号整数的三种表⽰⽅法均有符号位和数值位两部分,2进制序列中,最⾼位的1位是被当做符号位,剩余的都是数值位。符号位都是⽤0表⽰“正”,⽤1表⽰“负”。

正整数的原、反、补码都相同。
负整数的三种表⽰⽅法各不相同。

原码:直接将数值按照正负数的形式翻译成⼆进制得到的就是原码。

反码:将原码的符号位不变,其他位依次按位取反就可以得到反码。

补码:反码+1就得到补码。
补码得到原码也是可以使⽤:取反,+1的操作。

例如:

正数

负数

对于整形来说:数据存放在内存中其实存放的是补码,这是为什么呢?

在计算机系统中,数值⼀律⽤补码来表⽰和存储。原因在于,使⽤补码,可以将符号位和数值域统⼀处理;同时,加法和减法也可以统⼀处理(CPU只有加法器)此外,补码与原码相互转换,其运算过程是相同的,不需要额外的硬件电路。

注:源码,反码,补码的概念只适用于整型。

四、移位操作符

<< 左移操作符 
>> 右移操作符 

注:移位操作符的操作数只能是整数

(4.1)左移操作符

移位规则:左边抛弃、右边补0

具体演示:

(4.2)右移操作符

移位规则:⾸先右移运算分两种:
1.逻辑右移:左边⽤0填充,右边丢弃
2.算术右移:左边⽤原该值的符号位填充,右边丢弃

逻辑右移:

算数右移:

注意:对于移位运算符,不要移动负数位,这个是标准未定义的。

五、位操作符

位操作符有:

注:他们的操作数必须是整数。

这里直接用代码来介绍每个操作符:

按位与:

解释:

按位或:

解释:

按位异或:

解释:

按位取反:

解释:

六、单目操作符

单⽬操作符有这些:!、++、--、&、*、+、-、~ 、sizeof、(类型)

单⽬操作符的特点是只有⼀个操作数,在单⽬操作符中只有&和*没有介绍,这2个操作符,&是取地址,*是解引用。等学习指针时在详细介绍。

七、逗号表达式

逗号表达式,就是⽤逗号隔开的多个表达式。逗号表达式从左向右依次执⾏。整个表达式的结果是最后⼀个表达式的结果。基本格式为:

例子:

解释:

八、下标访问和函数调用操作符

(8.1)下标引用操作符  [ ]

操作数:⼀个数组名 + ⼀个索引值

例如:

(8.2)函数调用操作符 ()

接受⼀个或者多个操作数:第⼀个操作数是函数名,剩余的操作数就是传递给函数的参数。

例如:

九、操作符的优先级和结合性

(9.1)优先级

优先级指的是,如果⼀个表达式包含多个运算符,哪个运算符应该优先执⾏。各种运算符的优先级是不⼀样的。

例如:

上⾯⽰例中,表达式 3 + 4 * 5 ⾥⾯既有加法运算符( + ),⼜有乘法运算符( * )。由于乘法的优先级⾼于加法,所以会先计算 4 * 5 ,⽽不是先计算 3 + 4 。

运算符的优先级顺序很多,下⾯是部分运算符的优先级顺序(按照优先级从⾼到低排列),建议⼤概记住这些操作符的优先级就⾏,其他操作符在使⽤的时候查看下⾯表格就可以了。
• 圆括号( () )
• ⾃增运算符( ++ ),⾃减运算符( -- )
• 单⽬运算符( + 和 - )
• 乘法( * ),除法( / )
• 加法( + ),减法( - )
• 关系运算符( < 、 > 等)
• 赋值运算符( = )
由于圆括号的优先级最⾼,可以使⽤它改变其他运算符的优先级。

参考:C 运算符优先级 - cppreference.com

(9.2)结合性

如果两个运算符优先级相同,优先级没办法确定先计算哪个了,这时候就看结合性了,则根据运算符是左结合,还是右结合,决定执⾏顺序。⼤部分运算符是左结合(从左到右执⾏),少数运算符是右结合(从右到左执⾏),⽐如赋值运算符( = )。

例如:

上⾯⽰例中, * 和 / 的优先级相同,它们都是左结合运算符,所以从左到右执⾏,先计算 * ,再计算 /  。

十、表达式求值

(10.1)整型提升

C语⾔中整型算术运算总是⾄少以缺省整型类型的精度来进⾏的。为了获得这个精度,表达式中的字符和短整型操作数在使⽤之前被转换为普通整型,这种转换称为整型提升。

整型提升的意义:表达式的整型运算要在CPU的相应运算器件内执⾏,CPU内整型运算器(ALU)的操作数的字节⻓度⼀般就是int的字节⻓度,同时也是CPU的通⽤寄存器的⻓度。因此,即使两个char类型的相加,在CPU执⾏时实际上也要先转换为CPU内整型操作数的标准⻓度。通⽤CPU(general-purpose CPU)是难以直接实现两个8⽐特字节直接相加运算(虽然机器指令中可能有这种字节相加指令)。所以,表达式中各种⻓度可能⼩于int⻓度的整型值,都必须先转换为int或unsigned int,然后才能送⼊CPU去执⾏运算。

(10.1.1)如何进行整型提升

1. 有符号整数提升是按照变量的数据类型的符号位来提升的
2. ⽆符号整数提升,⾼位补0

负数的整形提升:

正数的整形提升:

运算时:

为什么会有这种情况呢?这是因为b和c的值被提升为普通整型,然后再执⾏加法运算。加法运算完成之后,结果将被截断,然后再存储于a中。截断是因为char类型存不下提升后的数据。如果这里直接打印计算结果,就不会发生截断。前者截断,后者不截断,导致结果发生改变。

注:截断就是取结果的后八位(对于char类型而言)。

(10.2)算数转换

如果某个操作符的各个操作数属于不同的类型,那么除⾮其中⼀个操作数的转换为另⼀个操作数的类型,否则操作就⽆法进⾏。下⾯的层次体系称为寻常算术转换。

如果某个操作数的类型在上⾯这个列表中排名靠后,那么⾸先要转换为另外⼀个操作数的类型后执⾏运算。

  • 59
    点赞
  • 52
    收藏
    觉得还不错? 一键收藏
  • 38
    评论
评论 38
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值