探秘C语言中整数的二进制表示:原码、反码、补码,以及大小端字节序的奥秘

news2024/10/2 16:25:00

在这里插入图片描述

本篇博客会讲解整数在内存中的存储形式,以及整数二进制的3种表示形式:原码、反码、补码,还有大小端的相关知识点。相信读完本篇博客,大家对内存的了解会上一个台阶。

注意:本篇博客讨论的是整数在内存中的存储,像char, short, int, long, long long的有符号和无符号类型存储的都是整数。

认识原码、反码、补码

对于整数,有原码、反码、补码之分,它们都是整数的二进制的表示形式。有符号整数的计算方式如下:

  1. 正整数的原码、反码、补码相同,都是整数直接转换成二进制的结果。
  2. 负整数的原码是把该数转换成二进制的结果,反码和补码需要通过原码计算出来。
  3. 负整数的反码的计算规则:原码的符号位不变,其他位按位取反得到反码。
  4. 负整数的补码的计算规则:反码+1得到补码。

下面来举2个例子,分别表示正整数和负整数。

正整数以10为例子:10作为十进制数,直接转换成二进制,得到的就是它的原码。又因为10是正整数,故原码、反码、补码相同。所以问题呢就转换成如何把十进制的10转换成二进制。

把十进制的10转换成二进制,得到的是1010。这是怎么来的呢?对于二进制的1010,从右往左数,最右边的0代表0×20,倒数第二位的1代表1×21,倒数第三位的0代表0×22,最左边的1代表1×23,所以二进制的1010就代表十进制的1×23+0×22+1×21+0×20=8+0+2+0=10。

严谨点来说:10的默认类型是int,一个int大小是4个字节,也就是32个bit位,应该填满32位,不够在高位补0,所以10的原码、反码、补码都是:00000000000000000000000000001010

再举个负数的例子:-10的原码、反码、补码是多少?对于负数,是“有符号数”,最高位表示符号位,用1表示负数,其余的位和10的二进制相同:10000000000000000000000000001010。这就是-10的原码。

-10作为一个负数,反码和补码需要通过原码来计算。反码的计算规则是,原码符号位不变,其他位按位取反,也就是说,最高位的1表示符号位,不变,其余位1变0,0变1。反码就是:11111111111111111111111111110101

补码的计算规则是:反码+1即是补码,所以-10的补码是:11111111111111111111111111110110

整数在内存中是以哪种形式存储的呢?事实上,整数在内存中存储的是二进制的补码,这一点是可以验证的,但是同时涉及到大小端的问题。

大小端字节序

大小端字节序,简称大小端,指的是,以字节为单位,内存中的数据是如何存储的。分为2种存储方式,分别是大端字节序和小端字节序,简称大端和小端。

  1. 大端字节序:把高位字节处的数据存储在低地址处,把低位字节处的数据存储在高地址处。
  2. 小端字节序:把高位字节处的数据存储在高地址处,把低位字节处的数据存储在低地址处。

这是什么意思呢?我在VS2022环境下验证一下:根据前面所讲,整数在内存中存储的是补码,我们来观察一下-10在内存中是如何存储的。

在这里插入图片描述
观察到,当左边是低地址,右边是高地址,即地址从左到右是由低到高变化的,此时内存中存储的是0x f6 ff ff ff。这是什么意思呢?

整数在内存中是以二进制的补码的形式存储的。前面计算过了,-10的补码是:11111111111111111111111111110110。下面我们把它转换成十六进制。

先4个为一组分开:1111 1111 1111 1111 1111 1111 1111 0110

然后分别转换每一组,其中1111是f,0110是6:0x fffffff6。一个十六进制位是4个bit位,2个十六进制位就是一个字节,把这个数以字节为单位分隔开,就是:0x ff ff ff f6

而内存中存储的却是:0x f6 ff ff ff,这是为什么呢?在VS环境下观察内存,从左到右,地址是由低到高变化的,也就是说,本来-10补码中的高位的ff存储在了高地址处,而-10补码中的低位的f6存放在低地址处,这就是“小端字节序”的存储方式,看起来像是数据“倒着放”。

整数在内存中的存储

整数在内存中的存储,是由以下2点决定的:

  1. 整数在内存中存储的是补码的二进制。
  2. 根据机器是“大端字节序”还是“小端字节序”来决定是“正着放”还是“倒着放”。

以上是数据“如何存”,那如何把一个数据“往外拿”呢?把存的过程倒过来就行了。

  1. 根据大小端字节序,把数据“正着”或者“倒着”拿出来。
  2. 把拿出来的补码转换成原码。

那如何把补码转换成原码呢?原码怎么转换成补码,把这个过程倒过来就行了,以下是如何把补码解析成有符号数的过程。

  1. 若符号位是0,则是正数,此时原码、反码都和补码相同。若符号位是1,则是负数,需要计算反码和原码。
  2. 补码-1得到反码。
  3. 反码的符号位不变,其他位按位取反得到原码。

注意:如果要解析成无符号数,最高位就不是符号位了,而是数据的一部分,此时一律看做正整数来处理。

举个例子,前面我们观察到小端机器下-10在内存中存储的是:0x f6 ff ff ff,由于小端是“倒着存”的,所以先正过来:0x ff ff ff f6,再转换成二进制:1111 1111 1111 1111 1111 1111 1111 0110,即11111111111111111111111111110110,这就是补码,由于符号位(最高位)是1,是一个负数,需要根据补码计算反码和原码。先把补码-1,得到反码:11111111111111111111111111110101,把反码符号位不变,其他位按位取反得到原码:10000000000000000000000000001010,再转换成十进制,就得到了-10。

为什么要存补码?

整数在内存中存储的是补码的二进制,这主要有2个优点。

先说第一个优点:使用补码来存储,统一了符号位和数值位。换句话说,符号位和数值为可以统一计算。在CPU中,只有加法器,所有的计算都要转换成加法。比如,计算1-1,先要转换成计算1+(-1),然后写出1和-1的补码:

1作为正数,原码、反码、补码相同,都是:00000000000000000000000000000001

-1作为负数,原码是:10000000000000000000000000000001,原码符号位不变,其他位按位取反后得到反码:11111111111111111111111111111110,反码+1得到补码:11111111111111111111111111111111

接下来计算1+(-1),也就是这样计算:

  00000000000000000000000000000001
+ 11111111111111111111111111111111
 100000000000000000000000000000000

由于整数只能存储32个bit为,最高位的1就丢了,只剩下00000000000000000000000000000000,得到的结果就是0。

根据以上的计算,有没有发现,我们根本没有管哪里是符号位,哪里是数值位,就是简单粗暴的把补码写出来,加起来,完事。CPU把内存中存储的补码拿出来,进行加法运算时,根本不用考虑哪里是符号位,那里是数值位,这就做到了把符号位和数值位统一处理。

在说第二个优点之前,还需要了解一个知识点:把补码转换成原码的另一种方式:先把补码的符号位不变,其他位按位取反,再+1,也能得到原码。比如:已知-10的补码:11111111111111111111111111110110,把补码的符号位不变,其他位按位取反得到:10000000000000000000000000001001,再+1得到:10000000000000000000000000001010,这样也得到了-10的原码。

有没有发现,如果用这种方式把补码转原码,那么就和原码转补码的方式相同了,都是“取反再+1”!所以,用补码来存储时,把补码转原码,以及原码转补码可以使用同一套硬件电路,降低了电路的复杂性。

总结

  1. 整数的2进制有3种表示形式,分别是原码、反码、补码。
  2. 正整数的原码、反码、补码相同,负整数的原码、反码、补码需要计算。把整数直接转换成2进制,就是原码。负整数的原码的符号位不变、其他位按位取反得到反码。负整数的反码+1得到补码。
  3. 从补码转原码,需要先看符号位,确定是正数还是负数,如果是正数,则原码、反码、补码相同。如果是负数,则可以先-1得到反码,再符号位不变,其他位按位取反得到原码;也可以先符号位不变,其他位按位取反,再+1得到原码。注意:无符号数一律按照正数来处理。
  4. 大小端字节序分为大端字节序和小端字节序。大端字节序指的是,把高位字节处的数据存储在低地址处,把低位字节处的数据存储在高地址处。小端字节序指的是,把高位字节处的数据存储在高地址处,把低位字节处的数据存储在低地址处。
  5. 内存中使用补码来存储整数有2个好处:统一处理符号位和数值位,以及使用同一套硬件电路来把原码和反码相互转换,降低电路复杂性。

感谢大家的阅读!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/484584.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

将mininet流量数据可视化至前端

目录 准备工作:将mininet流量数据输入数据库流量可视化 准备工作: 创建项目 django-admin startproject mininet_webpython manage.py runserver 0.0.0.0:8000init文件加上: settings改数据库,具体看上一篇 创建第一个app mo…

Windows安装mariadb,配置环境变量(保姆级教学)

软件下载地址:https://mariadb.com/downloads/ 1.双击下载好的软件 2.点击next 3.勾选我同意,点击next 4.这里那你可以设置你要安装的路径,也可以使用默认的,之后点击next 5.如图所示,设置完点击next 6.接下来就默…

如何让ChatGPT成为科研工作中的小助手?(附使用指南)

大家好,我是带我去滑雪! 从2022年年底发布叫ChatGPT的人工智能聊天机器人以来,逐渐强势进入了各行各业,一夜火爆全网,它使用自然语言处理技术来与用户进行交互和沟通,可以回答用户关于知识、娱乐、生活等方…

MySQL高级篇——MVCC多版本并发控制

导航: 【黑马Java笔记踩坑汇总】JavaSEJavaWebSSMSpringBoot瑞吉外卖SpringCloud黑马旅游谷粒商城学成在线MySQL高级篇设计模式牛客面试题 目录 1. 什么是MVCC 2. 快照读与当前读 2.1 快照读 2.2 当前读 3. MVCC三剑客 3.1 回顾隔离级别 3.2 隐藏字段、Undo…

组合导航卡尔曼滤波几个杂项

1.组合导航卡尔曼滤波噪声协方差矩阵调参 在组合导航卡尔曼滤波算法中,主要涉及两个噪声协方差矩阵,过程噪声协方差矩阵Q,测量噪声协方差矩阵R,具体来说: R表示测量噪声协方差,它是一个数值,这…

【Unity-UGUI控件全面解析】| Toggle 开关组件详解

🎬【Unity-UGUI控件全面解析】| Toggle 开关组件详解一、组件介绍二、组件属性面板三、代码操作组件四、组件常用方法示例4.1 监听开关事件五、组件相关扩展使用5.1 配合Toggle Group组使用💯总结🎬 博客主页:https://xiaoy.blog.csdn.net 🎥 本文由 呆呆敲代码的小Y …

JavaScript实现输入年份月份,判断该月份是多少天的代码

以下为实现输入年份月份,判断该月份是多少天的代码和运行截图 目录 前言 一、实现输入年份月份,判断该月份是多少天 1.1 运行流程及思想 1.2 代码段 1.3 JavaScript语句代码 1.4 运行截图 前言 1.若有选择,您可以在目录里进行快速查找…

康耐视Visionpro-视觉人机器视觉粉丝-千问之六十五解答

(2023年5月2日更,下次更新2023年10月1日-10月7日) Question0: 康耐视visionpro9.8/9.9-BeadInspect工具详细使用流程 原因分析或解决办法 康耐视visionpro9.8-BeadInspect工具详细使用流程 (qq.com) Question1: C#与visisionpro联合开发exe文件开机启动设置 原因分析…

项目实战笔记

前台:呈现给用户的视觉和基本操作 后台:用户浏览网页时候,用户看不到的后台数据 前面两者都包含前端和后端 技术选型要统一 后端 接入层(control): 接受请求,获取传递过来的参数调用服务层…

unityt光线透射目标

介绍 在Unity中,光线透射目标通常指的是在场景中放置的一些物体,用于模拟光线从一个物体透过到另一个物体的效果。canvas子物体组件中,勾不勾选“光线透射目标”有什么区别? 方法 在Canvas子物体组件中勾选“光线透射目标”时&…

jqGrid之单选

jqGrid不支持单选,本来也想借鉴网上的做法,通过对checkbox的处理来间接实现。但在实践中却发现设置了multiSelect后,checkbox并没有显示出来,怀疑是跟页面的其它前端框架如bootstrap有冲突,索性换个思路来实现&#xf…

C语言的那些少见的事1

目录 前言: 1.变量名称的意义 2.卖函数库的知识 3.变长数组 4.清空输入缓冲区 5.左值和右值 6.for控制语句中定义变量 7.刷题中浮点数的精度问题 8.C语言提供C的标准和C库的约定 9.extern声明外部符号 ❤博主CSDN:啊苏要学习 ▶专栏分类:C语言…

飞书接入ChatGPT - 将ChatGPT集成到飞书机器人,直接拉满效率

文章目录 前言环境列表视频教程1.飞书设置2.克隆feishu-chatgpt项目3.配置config.yaml文件4.运行feishu-chatgpt项目5.安装cpolar内网穿透6.固定公网地址7.机器人权限配置8.创建版本9.创建测试企业10. 机器人测试 转载自远控源码文章:飞书接入ChatGPT - 将ChatGPT集…

java基础知识——26.反射

这篇文章我们来讲一下java的代理与反射,这是很重要的一部分内容。 目录 1.什么是反射 2.获取class对象的三种方式 3.反射获取构造方法 4.利用反射来获取成员变量 5.利用反射来获取成员方法 6.反射的作用 7.反射小结 1.什么是反射 首先,我们来看…

研报精选230503

目录 【个股230503华西证券_山鹰国际】盈利能力环比改善,消费复苏静待花开 【个股230503华西证券_太平鸟】控折扣下毛利率历史新高,关注Q3货品改善 【个股230503中银证券_南京银行】息差企稳回升 【个股230503华西证券_山鹰国际】盈利能力环比改善&#…

No.046<软考>《(高项)备考大全》【专项2】《案例分析 - 计算题(1)》

《(高项)备考大全》【专项2】《案例分析 - 计算题》 1 题型全部概况2 时间管理2.1 关键路径法 CMP2.1.1 原理2.1.2 关键路径的基本问题2.1.3 题目7、题目6 - 正推、反推8、题目7 2.2 PERT(计划评审技术)2.3 活动排序网络图 3 成本…

安卓系统下的截屏和录屏

可以抓取手机屏幕画面(屏幕截图),也可以录制屏幕画面视频。拍摄屏幕后,可以查看、编辑和分享所拍的图片或视频。 抓取屏幕截图 打开要抓取的屏幕。视手机情况执行下列一个操作,3种方法看你手机有效的: 同…

linux【多线程】基于环形队列(RingQueue)的生产消费模型

基于环形队列RingQueue的生产消费模型 一、引入二、信号量2.1 信号量概念2.2 信号量PV操作2.3 POSIX信号量接口 三、基于环形队列(RingQueue)的生产消费模型3.1 设计思路3.2 结构设计图3.3 单生产单消费代码实现 四、多生产多消费情形五、小结5.1 多生产多消费的意义5.2 条件变…

docker容器技术

什么是docker Docker 使用 Google 公司推出的 Go 语言 进行开发实现,基于 Linux 内核的 cgroup,namespace,以及 OverlayFS 类的 Union FS 等技术,对进程进行封装隔离,属于 操作系统层面的虚拟化技术。由于隔离的进程独…