c语言 与0xff,带你在过程中理解&0xff[通俗易懂]

c语言 与0xff,带你在过程中理解&0xff[通俗易懂]在写大作业的时候,一开始对&0xff的操作一直处于疑惑状态.。byte[i]是8位二进制,0xff转化成8位二进制就是11111111,那么byte[i]&0xff不是还是byte[i]本身吗?Areyoukiddingme?对于这个问题,我在网上看到一个demo很有趣:#includeintmain(void){charbyte=-127;inta;…

大家好,又见面了,我是你们的朋友全栈君。

在写大作业的时候,一开始对 &0xff 的操作一直处于疑惑状态.。

byte[i] 是8位二进制,0xff 转化成8位二进制就是 11111111,那么byte[i] & 0xff不是还是byte[i]本身吗?

Are you kidding me?

583a99b88c81c6c348bd0907d1f47eea.png

对于这个问题,我在网上看到一个demo很有趣:

#include

int main(void)

{

char byte = -127;

int a;

a = byte;

printf(“%d\n”, a);

a = byte & 0xff;

printf(“%d”, a);

return 0;

}

然而~

8d6bd83ff6e3779b95c709ddee2b99b2.png

事情开始有趣了起来…

为什么加了 &0xff 反而不对了呢??

我们先来复习一下补数的概念:

对于正数(00000001)原码来说,首位表示符号位,反码与补码都是本身

对于负数(100000001)原码来说,反码是对原码除了符号位之外作取反运算即(111111110)然后作+1运算即(111111111)

再介绍一下符号扩充的概念:

以8位二进制数为例,符号扩充就是指在保持当前值不变的前提下将其转化为16位和32位的二进制数。规则就是,不管是正数还是用补码表示的负数,都只需要用符号位的值(0或1)来填充高位即可。

进入正题:

当将-127赋值给byte的时候,byte作为一个char类型,其计算机存储的补码是10000001(8位)[在计算机中,负数都是以补码形式储存]。

a = byte;//-127

第一次将 byte 作为int类型向控制台输出的时候,编译器作了一个符号扩充的处理,因为int类型是32位二进制数,所以byte扩充后的补码就是1111111111111111111111111 10000001(32位),这个32位二进制补码表示的十进制数也是-127.这说明符号扩充并不会影响当前对应的十进制数的值。

这是一个很好的性质,但是我们将char类型进行int类型的转化的时候,目的并不仅仅是要保证对应十进制数的不变性。比如说我们这次的大作业,是要将4个char类型转化成1个int类型,这就需要保证二进制补码的一致性,也就是4个char类型所对应的二进制01序列原封不动地作为一个int类型的4个字节(高八位 中八位 中八位 低八位)的二进制序列。

a = byte & 0xff;//129

那我们第二次进行赋值的时候为什么会改变a的值呢?

我们来具体分析一下:

之前介绍了符号扩充的概念,当byte(10000001)要转化为int的时候,高的24位必然会补1。这样,其二进制补码其实就已经改变了。而&0xff(11111111)可以将byte变int时,高的24位设置为0,低8位保持byte的原样。

那可能会有同学疑惑:为什么0xff 高的24位不会同样补1呢?这是因为 0xff或者0xFF本身就是一个int的字面常量,自身就是32位长的,所以不会进行符号扩充。

当然,在我想保证二进制补码一致性的时候,二进制数所对应十进制数自然也就发生变化啦。这是无法兼得滴~

来看看代码的具体实现:

a = byte & 0xff;//129

byte & 0xff = 111111111111111111111111110000001&11111111

=000000000000000000000000 10000001

这个二进制数位权加一下就是129啦

这很好的解释了为什么加了 &0xff 结果反而出错了:

因为这个时候的程序保证了二进制补码的一致性而不是对应十进制数的一致性。

最后

咱们回到刚开始大作业转换算法的问题。我们是想保证二进制补码的一致性,所以要先对byte进行 &0xff 操作。当然假如byte的符号位位0 ,那么 &0xff 就没有什么意义。但是当byte的符号位为1的时候,你就需要掂量一下了,因为你最后是要和 value进行 逻辑或 操作的(很大概率你辛辛苦苦获得的二进制序列可就全部变成1了哦)!!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/150811.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 查看本机上的端口使用情况netstat -an「建议收藏」

    查看本机上的端口使用情况netstat -an「建议收藏」1.查找本机上的端口使用情况netstat-an2.查找指定端口的使用情况C:\Windows\System32>netstat-ano|find"8002"

    2022年7月2日
    45
  • goland 2021.12激活【最新永久激活】

    (goland 2021.12激活)本文适用于JetBrains家族所有ide,包括IntelliJidea,phpstorm,webstorm,pycharm,datagrip等。https://javaforall.net/100143.htmlIntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,上面是详细链接哦~3…

    2022年3月30日
    244
  • maven打包时跳过测试「建议收藏」

    maven打包时跳过测试「建议收藏」开发记录有一个maven项目,我clone一下最新的代码。准备打包(mavenpackage),没想到在执行到TEST阶段报错。百思不得其解,决定跳过测试去打包,然后部署。Failedtoexecutegoalorg.apache.maven.plugins:maven-surefire-plugin:2.12.4:test在运行mvninstall或mvnpackag…

    2022年6月9日
    32
  • 美化包软件_彩色音量进度条插件下载

    美化包软件_彩色音量进度条插件下载前言在我们进行自动化测试的时候,用例往往是成百上千,执行的时间是几十分钟或者是小时级别。有时,我们在调试那么多用例的时候,不知道执行到什么程度了,而pytest-sugar插件能很好解决我们的痛点。

    2022年7月28日
    5
  • pycharm git使用_pycharm上传github

    pycharm git使用_pycharm上传githubpycharm操作git一、git安装和使用​ 安装操作:https://www.cnblogs.com/ximiaomiao/p/7140456.html1.如何使用git将本地代码上传到远程仓库初始化gitinit查看当前仓库状态gitstatus将项目的文件添加到仓库中gitadd<文件名>gitadd.(上传所有文件)将add的文件commit到仓库gitcommit-m”备注”将本地仓库关联到远程仓库gi

    2022年8月25日
    6
  • 高等数学解题神器app_ubuntu cp命令

    高等数学解题神器app_ubuntu cp命令XSS在chrome上,需要先关闭xss保护反射型low对输入未做过滤$data=no_check($data);输入&lt;script&gt;alert(document.cookie)&lt;/script&gt;middle输入校验functionxss_check_4($data){//addsla…

    2022年9月23日
    1

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号