数据的存储--深度解剖数据在内存中的存储(上)

简介: 数据的存储--深度解剖数据在内存中的存储(上)

本章重点

1.数据类型详细介绍

2.整形在内存中的存储:原码、反码、补码

3.大小端字节序介绍及判断

4.浮点型在数据中的存储解析

一 数据类型介绍

前面我们已经学习了一些基本的数据类型。

long long这种类型在C99中才有,所以在新的编译器中才有这种类型。VC6.0中就不支持这种类型

int 在早期16位平台上是2个字节,现在的32位平台和64位平台都是4个字节

C语言规定sizeof(long)>=sizeof(int)                       long long 8个

在新的编译器里面还有long double     在vs2019 也是8个字节

1.1 类型的基本分类

整形家族

方括号[] 说明可以省略,也可以不省略

char

     unsigned char

     signed char

short

     unsigned short [int]

     signed short [int]

int

     unsigned int (无符号整型)      

     signed int (有符号整形)

long

     unsigned long [int]

     signed long [int]

字符(char)的ASCII码值是整形,所以char也归到整形家族

long long也属于整形家族

无符号整型用%u打印

平常中的int(可以放正数,也可以放负数) 就是signed int(signed int 等价于 int),unsigned int 无论里面放的是正数还是负数,都会转化为相应的正整数来计算。

signed short 等价于 short

signed long等价于 long

char 等价于signed char 还是 unsigned char 是取决于编译器实现的,常见的编译器是等价于signed char ,VS2019 char 等价于signed char 。

浮点数家族

float

double

float 单精度 double 双精度,它的精度更加高

构造类型

数组类型

结构体类型 struct

枚举类型 enum

联合类型 union

构造类型,又叫自定义类型,例如 int arr[5] 和 int arr[7] 他们两个的类型分别是 int [5] int [7], 通过自己的定义而发生改变。

指针类型

int *pi;

char *pc;

float* pf;

void* pv;

空类型

void 表示空类型(无类型)

通常应用于函数的返回类型、函数的参数、指针类型

函数参数void说明不需要传入参数,void*p,说明p可以是任何类型的指针类型

二 整形在内存中的存储

变量的创建是要在内存中开辟空间的。空间的大小是根据不同的类型决定的。

注意:在调试器的内存窗口中,展示的是16进制,但是内存中储存的是二进制(为了方便展示)

2.1 原码、反码、补码

计算机的整数有三种表示方法,即原码、反码、补码。

三种表示方法,均有符号位和数值位两部分。符号位都是用0表示“正”,用1表示“负”

负整数的三种表示方法各不相同

原码

直接将二进制按照正负数的形式翻译成二进制就可以。

反码

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

补码                                                                                                                                                    反码+1

正数的原、反、补码都相同

0的原码、反码、补码都是000000000 00000000 00000000 00000000

对于整形来说:数据存放内存中其实存放的是补码:原因是:在计算机系统中,数值一律用补码来表示和存储。原因在于,使用补码,可以将符号位和数值域统一处理;原码相加会出现错误,补码就不会。同时,加法和减法也可以统一处理(CPU只有加法器)(计算机中只有加法器,减法是模拟出来的)此外,补码与原码相互转换,其运算过程是相同的,不需要额外的硬件电路

四位二进制表示的最大数字是16,所以两位16进制表示一个字节,一个字节=8位在编译器内存显示器中10,0a 00 00 00,数据的低位放在低地址中,这就是下面要讲述的大小端。

2.2 大小端介绍

什么大端小端:

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

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

char 类型是没有大小端的,因为char就一个字节,大小端是以字节为单位的。

在这里就可以非常明显的看见数据的低位放在内存的低地址中。

为什么有大端和小端:

为什么会有大小端模式之分呢?这是因为在计算机系统中,我们是以字节为单位的,每个地址单元

都对应着一个字节,一个字节为8bit。但是在C语言中除了8 bit的char之外,还有16 bit的short型,32 bit的long型(要看具体的编译器),另外,对于位数大于8位的处理器,例如16位或者32位的处理器,由于寄存器宽度大于一个字节,那么必然存在着一个如何将多个字节安排的问题。因此就

导致了大端存储模式和小端存储模式。

例如:一个 16bit 的 short 型 x ,在内存中的地址为 0x0010 , x 的值为 0x1122 ,那么 0x11 为

高字节, 0x22 为低字节。对于大端模式,就将 0x11 放在低地址中,即 0x0010 中, 0x22 放在高地址中,即 0x0011 中。                                                                                                                     小端模式,刚好相反。我们常用的 X86 结构是小端模式,而 KEIL C51 则为大端模式。很多的ARM,DSP都为小端模式。有些ARM处理器还可以由硬件来选择是大端模式还是小端模式。

设计一个程序,判断当前机器的字节位。

a = 1; 0x 00 00 00 01 (大端);0x 01 00 00 00 (小端),比较第一个地址的是否等于1,即可,仅仅想拿第一个字节的地址,那么解引用的时候用char即可。

代码1展示:

1. #include <stdio.h>
2. int main()
3. {
4.  int a = 1;
5.  char* pa = (char*)&a;//强制类型转换
6.  if (*pa == 1)
7.  {
8.    printf("小端\n");
9.  }
10.   else
11.   {
12.     printf("大端\n");
13.   }
14.   return 0;
15. }

代码2展示:(函数以及简化)

1. #include <stdio.h>
2. int check_sys()
3. {
4.  int a = 1;
5.  return *(char*)&a;
6. }
7. int main()
8. {
9.  int a = check_sys();
10.   if (a == 1)
11.   {
12.     printf("小端\n");
13.   }
14.   else
15.   {
16.     printf("大端");
17.   }
18.   return 0;
19. }


相关文章
|
2月前
|
存储 编译器 数据处理
C 语言结构体与位域:高效数据组织与内存优化
C语言中的结构体与位域是实现高效数据组织和内存优化的重要工具。结构体允许将不同类型的数据组合成一个整体,而位域则进一步允许对结构体成员的位进行精细控制,以节省内存空间。两者结合使用,可在嵌入式系统等资源受限环境中发挥巨大作用。
78 11
|
3月前
|
监控 算法 应用服务中间件
“四两拨千斤” —— 1.2MB 数据如何吃掉 10GB 内存
一个特殊请求引发服务器内存用量暴涨进而导致进程 OOM 的惨案。
|
3月前
|
存储 C语言
数据在内存中的存储方式
本文介绍了计算机中整数和浮点数的存储方式,包括整数的原码、反码、补码,以及浮点数的IEEE754标准存储格式。同时,探讨了大小端字节序的概念及其判断方法,通过实例代码展示了这些概念的实际应用。
163 1
|
3月前
|
存储
共用体在内存中如何存储数据
共用体(Union)在内存中为所有成员分配同一段内存空间,大小等于最大成员所需的空间。这意味着所有成员共享同一块内存,但同一时间只能存储其中一个成员的数据,无法同时保存多个成员的值。
|
2月前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
434 1
|
1月前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
2月前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80
|
2月前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
27 3
|
2月前
|
存储 缓存 监控
Elasticsearch集群JVM调优堆外内存
Elasticsearch集群JVM调优堆外内存
59 1
|
2月前
|
Arthas 监控 Java
JVM进阶调优系列(9)大厂面试官:内存溢出几种?能否现场演示一下?| 面试就那点事
本文介绍了JVM内存溢出(OOM)的四种类型:堆内存、栈内存、元数据区和直接内存溢出。每种类型通过示例代码演示了如何触发OOM,并分析了其原因。文章还提供了如何使用JVM命令工具(如jmap、jhat、GCeasy、Arthas等)分析和定位内存溢出问题的方法。最后,强调了合理设置JVM参数和及时回收内存的重要性。