数据在内存中的存储

目录

数据类型

大小端 

判断大小端 

练习

1

 2

浮点数在内存中储存

 存M

存E

 取E


数据类型

整形家族:

char
 unsigned char
 signed char
short
 unsigned short [int]
 signed short [int]
int
 unsigned int
 signed int
long
 unsigned long [int]
 signed long [int]

其中signed类型和不写signed是一种类型。

char被归入整形是因为它本质储存ascii码值为二进制。

浮点型 

float

double

构造类型 

> 数组类型
> 结构体类型 struct
> 枚举类型 enum
> 联合类型 union

 指针类型

int *pi;
char *pc;
float* pf;
void* pv;

空类型 

void 表示空类型(无类型)
通常应用于函数的返回类型、函数的参数、指针类型。

大小端 

大端(存储)模式,是指数据的低位保存在内存的高地址中,而数据的高位,保存在内存的低地址中;
小端(存储)模式,是指数据的低位保存在内存的低地址中,而数据的高位,保存在内存的高地址中。

其实可以理解为我们站队时按从低到高还是从高到低排序,每个内存单元以字节为单位,超过一个字节就会出现如何安排字节的问题,所以就有了大小端。我们也可以通过调试观察编译器的存放模式。

 用16进制我们可以看到编译器采用的是小端字节序的排列方式。

判断大小端 

要证明是大端还是小端,我们可以创建一个字节数大于1的变量(这样才有大小端),然后对首字节进行判断即可。访问一个字节我们可以用char*。

int check_sys()
{int a = 1;if (*(char*)&a == 1)return 1;elsereturn 0;
}
//int check_sys()
//{
//	int a = 1;
//	return *(char*)&a;
//}
int main()
{if(check_sys() == 1)printf("小端\n");elseprintf("大端\n");return 0;
}

练习

1

我们先来看看这样一段代码:

 char a = -128;printf("%u", a);

请问这段代码会输出什么结果呢?我们得进行补码运算得到它的二进制码

-128补码: 10000000(转化成int后截断放入char的二进制码)

unsigned int类型输出a,发生整形提升(补符号位):11111111111111111111111110000000,由于最高位是1,要转化成原码,得到一个很大的正数,我们可以验证一下。

 2

有这样一段代码:

        char a[1000];int i;for (i = 0; i < 1000; i++){a[i] = -1 - i;}printf("%d", strlen(a)); 

我们先来分析for循环内部,有1000个元素,元素取值从-1到-1000,但对于char类型来说,它的取值范围为-128~127, 当超过-128时,数据会跳到127去,也就是从最小变到最大,反过来也成立。可以理解成下面的循环:

1

也就是说,该程序在进行128+127 = 255次循环后将遇到\0,所以会输出255。 

 同样,这段代码也会出现相同的错误,造成死循环:

        unsigned char i = 0;for (i = 0; i <= 255; i++){printf("hello world\n");}

浮点数在内存中储存

浮点数用于储存小数,但在储存一些数比如说无理数时可能会丢失精度。我们可以在limits.h查找整形家族的取值范围和从float.h中查找浮点型家族的取值范围。

我们先来看一段奇怪的代码: 

int n = 9;
float* pFloat = (float*)&n;
printf("n的值为:%d\n", n);	
printf("*pFloat的值为:%f\n", *pFloat);*pFloat = 9.0;printf("num的值为:%d\n", n);		
printf("*pFloat的值为:%f\n", *pFloat);

你们可能以为结果为9,9.000000,9,9.000000,但真实结果却是这样的:

        好,我们来看下浮点数的储存方式——

根据IEE754标准,我们可以将二进制浮点数V这样表示:

  • (-1)^S * M * 2^E
  • (-1)^S表示符号位,当S=0,V为正数;当S=1,V为负数。
  • M表示有效数字,大于等于1,小于2。
  • 2^E表示指数位

例:5.5转化成二进制:

>>>2^2 + 2^0 +2^-1 = 101.1 (小数点后表示2^-1,2^-2....)

转化成(-1)^S * M * 2^E的形式:

>>>(-1)^0 * 1.011 * 2^2

现在我们来看内存中的储存模型:

 存M

M的取值范围规定为1<=M<2,计算机保存M时,默认保存的第一个数为1,所以可以省去使其能储存24个bit位

存E

        首先,E为一个无符号整数,这意味着,如果E为8位,它的取值范围为0~255;如果E为11位,它的取值范围为0~2047。但是,科学计数法中的E是可以出现负数的,所以IEEE754规定,存入内存时E的真实值必须再加上一个中间数,对于8位的E,这个中间数是127;对于11位的E,这个中间数为1023

 取E

复杂的是,取E分为3种情况:

  1. E不全为0且不全为1
  2. E全为0
  3. E全为1

        第一种情况取出时减去127即可得可。

        第二种情况我们通过分析,全为0时E = -127, 已经是一个极其小的数,还原时不再加上第一位的1,而是还原成0.xxxx的小数(1-127或1-1023为E的真实值),这样做是为了表示±0和接近0的的很小的数字。

        第三种情况就表示一个很大的数字,表示±无穷大

对于刚才的(-1)^0 * 1.011 * 2^2,我们可以这样存储:
>>>0 10000001 011000000000000000000000

转化成16进制:

>>>40b00000

内存中存储(小端字节序):
>>>00 00 b0 40

现在回到最初的问题代码:

float* pFloat = (float*)&n;

通过这段代码通过创建一个float指针接收了被强转的int类型的地址&n,使其看作IEE754标准存储。

将pFloat按照%f输出时9的补码为:

0 00000000 00000000000000000001001

转换成数字:

(-1)^0 * 0.00000000000000000001001 * 2^-126
//打印0.000000

第二次将9.0这个浮点型放入pfloat,经过下面的变换:

    9.01001.01.001 * 2^3(-1)^0 * 1.001 *2^3

得到内存空间:

01000001000100000000000000000000

将其按整形补码(原反补相同)输出得到:1,091,567,616

按浮点形输出得到:9.000000

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

关于浅克隆和深克隆入门理解

浅克隆:需要类实现Cloneable,并重写clone()方法 一般在重写clone()方法时,将返回值类型强转为自己类,避免每次克隆之后需要强转 public class Test {public static void main(String[] args) throws CloneNotSupportedException {A a1new A();A a2 a1.clone();//克隆之后 a1…

实时测试工具 Visual Studio 扩展 NCrunch 4.18 Crack

NCrunch Visual Studio 扩展 .NET 的终极实时测试工具 在编码时查看实时测试结果和内联指标。 下载v4.18 发布于 2023 年 7 月 17 日 跳过视频至&#xff1a; 代码覆盖率 指标 分布式处理 配置 发动机模式 Visual Studio 自动并发测试 NCrunch 是一个完全自动化的测试扩展&a…

C++day7

仿照vector手动实现自己的myVector&#xff0c;实现二倍扩容功能 #include <iostream>using namespace std;template<typename T> class my_vector {int size;//可存储的容量大小int num;//当前存储的元素个数T* data;//存储数据的空间地址public://无参构造函数m…

怎么获取别人店铺的商品呢?

jd.item_search_shop(获得店铺的所有商品) 为了进行电商平台 的API开发&#xff0c;首先我们需要做下面几件事情。 1&#xff09;开发者注册一个账号 2&#xff09;然后为每个JD应用注册一个应用程序键&#xff08;App Key) 。 3&#xff09;下载JDAPI的SDK并掌握基本的API…

HSRP(热备份路由选择协议)的概念,原理与配置实验

作者&#xff1a;Insist-- 个人主页&#xff1a;insist--个人主页 梦想从未散场&#xff0c;传奇永不落幕&#xff0c;持续更新优质网络知识、Python知识、Linux知识以及各种小技巧&#xff0c;愿你我共同在CSDN进步 目录 一、了解HSRP协议 1. 什么是HSRP协议 2、HSRP协议的…

概率论与数理统计学习笔记(7)——全概率公式与贝叶斯公式

目录 1. 背景2. 全概率公式3. 贝叶斯公式 1. 背景 下图是本文的背景内容&#xff0c;小B休闲时间有80%的概率玩手机游戏&#xff0c;有20%的概率玩电脑游戏。这两个游戏都有抽卡环节&#xff0c;其中手游抽到金卡的概率为5%&#xff0c;端游抽到金卡的概率为15%。已知小B这天抽…

算法分析与设计编程题 递归与分治策略

棋盘覆盖 题目描述 解题代码 // para: 棋盘&#xff0c;行偏移&#xff0c;列偏移&#xff0c;特殊行&#xff0c;特殊列 void dividedCovering(vector<vector<int>>& chessBoard, int dr, int dc, int sr, int sc, int size) {if (size 1) return;size / 2…

华为OD机试 - 字符串加密(Java 2023 B卷 100分)

目录 专栏导读一、题目描述二、输入描述三、输出描述四、解题思路五、Java算法源码六、效果展示1、输入2、输出3、说明 华为OD机试 2023B卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff09;真题&#xff08;A卷B卷&#…

算法分析与设计编程题 贪心算法

活动安排问题 题目描述 解题代码 vector<bool> greedySelector(vector<vector<int>>& intervals) {int n intervals.size();// 将活动区间按结束时间的从小到大排序auto cmp [](vector<int>& interval1, vector<int>& interval2…

【LeetCode-中等题】78. 子集

文章目录 组合并集问题汇总&#xff1a;题目方法一&#xff1a;动态规划方法二&#xff1a;递归加回溯(关键----startIndex) 组合并集问题汇总&#xff1a; 1、子集去重版本 2、组合非去重版本 3、组合去重版本 题目 注意&#xff1a;这里的nums数组里面的元素是各不相同的&a…

SpringBoot2.0(Lombok,SpringBoot统一返回封装)

目录 一&#xff0c;Lombok简介二&#xff0c;添加依赖三&#xff0c;Springboot统一返回封装3.1&#xff0c;创建一个工具包util和一个JsonData类 四&#xff0c;示例实体类五&#xff0c;Data注解 一&#xff0c;Lombok简介 ​ java工程中&#xff0c;我们要创建很多的java B…

SpringCloud(17~21章):Alibaba入门简介、Nacos服务注册和配置中心、Sentinel实现熔断与限流、Seata处理分布式事务

17 SpringCloud Alibaba入门简介 17.1 why会出现SpringCloud alibaba Spring Cloud Netflix项目进入维护模式 https://spring.io/blog/2018/12/12/spring-cloud-greenwich-rc1-available-now 说明 Spring Cloud Netflix Projects Entering Maintenance Mode 什么是维护模…