C语言位操作

本文深入解析了C语言中的位操作,包括位与(&)、位或(|)、位异或(^)、位取反(~)、左移(<<)和右移(>>)。通过真值表阐述了它们的运算规则,并通过实例展示了如何进行位运算。同时,讨论了位操作与逻辑操作的区别,以及位操作在寄存器操作中的应用,如清零、置1和取反。此外,还介绍了位移位在数值计算和负数处理中的实现细节。
摘要由CSDN通过智能技术生成

C语言位操作

一、常用位操作符

位与&无符号数有符号数
位或|逻辑左移<<算数左移<<
位异或^逻辑右移>>算数右移>>
位取反~

1、位与

真值表:

&01
000
101

3 & 5 = ?

​ 0b0011 (3)

& 0b0101 (5)


= 0b0001 (1)

拓展:&(按位与)与&&(逻辑与)的区别

逻辑与是将运算的俩个数看做一个整体,而整体结果如果位0,则该数被定义成逻辑假(0);如果该数不为0(不管是正的还是负的),则被定义为逻辑真(1)。

3 && 5 = 1

3 && 0 = 0

3 && -5 = 1

2、位或(|)

真值表:

|01
001
111

3 | 5 = ?

​ 0b0011 (3)

| 0b0101 (5)


= 0b0111 (7)

拓展:&(按位或)与&&(逻辑或)的区别

逻辑或是将运算的俩个数看做一个整体,而整体结果如果位0,则该数被定义成逻辑假(0);如果该数不为0(不管是正的还是负的),则被定义为逻辑真(1)。

3 || 5 = 1
    
0 || 0 = 0
 
3 || -5 = 1

3、位取反(~)

将操作数的二进制数的位逐个按位取反(1变0,0变1)。

真值表如下:

~1 = 0
~0 = 1
~10 = ?
0b1010 (10) 01010 (10)

​ 0b0101 (5) 10101

​ 11011(-11)

拓展:~(位取反)与!(非)的区别

!(非)是将操作数整体看成一个整体,而这个整体如果是0,则该数被定义成逻辑假(0);如果该数不为0,(不管是正还是负的),则被定义成逻辑真(1)。

上面是正常的计算取反的方法,但是在计算机中并不是这样的,需要使用下面的方法。

这里先说一下二进制在内存的存储:二进制数在内存中以补码的形式存储

另外,正数的原码、补码和反码都相同

在计算机中,二进制数在内存中以补码的形式存储,因此要求出其原来的值,就需要在对该数求其补码。正数的原码补码,反码都相同。

从负数的原码求反码和补码解题方法和步骤

(1)保持符号位的1不变,将数字部分的每一位求反(1改为0,0改为1),就得到了反码。

(2)在反码的末位上加1,即得到补码。

4、位异或(^)

俩个数如果结果不等,则其结果为1,相等为0.

真值表如下:

^01
001
110

3 ^ 5 = ?

​ 0b0011 (3)

^ 0b0101 (5)

= 0b0110 (6)

5、左移位(<<)

左移位就是将一个操作数的各二进制位全部左移若干位,左边移出去的二进制位丢弃,右边突出的二进制位补0。

5 << 2 =?

0b 0000 0101 (5)

0b 0001 0100 (20)

每进行一次左移位,就是将原来的数乘2。 5x2x2=20

6、右移位(>>)

右移是将一个操作数的各个二进制位全部右移若干位,左边的二进制补0或补1,(如果操作数是无符号数或有符号整数就补0,如果是有符号负数就补1),右边的二进制位丢弃。

负数的存储是以补码的形式存储的,移位是对其负数对应的补码进行的移位,因此原来的数还要进行求原码。

-5 >> 2 = -2

​ 0b1111 1011 (-5)

有移一位 :0b11111101 (-3)

有移俩位: 0b11111110 (-2)

二、位操作与寄存器

1、寄存器特定位清零用&

如果希望将一个寄存器的某些特定位变成0而不影响其他位,可以构造一个合适的1和0组成的数,和这个寄存器原来的值进行位与操作,就可以将特定位清零。

假设原来32位的寄存器REG的值为0xAAAAAAAA 我们希望将bit9~bit16置为0而其他位不变,将这个数与0X0000 FF00进行位与即可。

REG & = 0x0000 FF00

2、寄存器特定位置1用|

如果希望将一个寄存器的某些特定位变成1而不影响其他位,可以构造一个合适的1和0组成的数,和这个寄存器原来的值进行位或操作,就可以将特定位置1。

假设原来32位的寄存器REG的值为0xAAAA00AA 我们希望将bit9~bit16置为1而其他位不变,将这个数与0X0000 FF00进行位或即可。

REG | = 0x0000 FF00

3、寄存器特定位取反用~

如果希望将一个寄存器的某些特定位0变成1,1变成0,,即取反而不影响其他位,可以构造一个合适的1和0组成的数,和这个寄存器原来的值进行异或操作,就可以将特定位清零。

假设原来32位的寄存器REG的值为0xAAAAAAAA 我们希望将bit9~bit16取反而其他位不变,将这个数与0X0000 FF00进行位异或即可。

REG ^ = 0x0000 FF00

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值