前言
数据在内存中是如何存储的呢
1.数据类型介绍
基本内置类型
整型:
C数据类型 | 最小值 | 最大值 |
---|---|---|
[unsigned] char | − 2 7 -2^7 −27 | 2 7 − 1 2^7-1 27−1 |
unsigned char | 0 | 2 8 − 1 2^8-1 28−1 |
short | − 2 15 -2^{15} −215 | 2 15 − 1 2^{15}-1 215−1 |
unsigned short | 0 | 2 16 − 1 2^{16}-1 216−1 |
int | − 2 31 -2^{31} −231 | 2 31 − 1 2^{31}-1 231−1 |
unsigned int | 0 | 2 32 − 1 2^{32}-1 232−1 |
long | − 2 63 -2^{63} −263 | 2 63 − 1 2^{63}-1 263−1 |
unsigned long | 0 | 2 64 − 1 2^{64}-1 264−1 |
32位程序c语言整型类型典型取值范围
浮点型:
C数据类型 | 位数 |
---|---|
float | 32 |
double | 64 |
构造类型:
C数据类型 | 关键字 |
---|---|
数组类型 | |
结构体类型 | struct |
枚举类型 | enum |
联合体类型 | union |
指针类型:
int *pi
char *pc
float *pf
void *pv
空类型:
void 表示空类型
通常用于函数返回类型,函数函数,指针类型
2.整型在内存中的存储
变量的创建需要在内存开辟空间,空间大小取决于数据的类型
那么数据在内存中是如何存储的呢?
例如:
int a = 10;
int b = -20;
它们被分配了四个字节的空间,如何存储?
原码、反码、补码
计算机中的整数有三种2进制表示方法,即原码、反码和补码。
三种表示方法均有符号位和数值位两部分,符号位用0表示正,用1表示负
正整数的原、反、补码均相同。
负整数的三种表示方法各不相同。
原码
直接将数值翻译成二进制即得到原码
反码
将原码符号位不变,其他位按位取反即得到反码
补码
反码+1即得到补码
举个例子
通过vs调试可以看到变量a和b在内存是如何存储的,由于a的原反补相同,看不出变量在内存如何存储,但通过b可以大概看出内存中存储变量的补码,但你会发现补码和内存中的数据有点不太一样,下面我将回答这一问题。
大、小端介绍
什么是大端小端
大端(存储)模式,是指数据的低位保存在内存高地址中,而数据的高位,保存在内存的低地址中。
小端(存储)模式,是指数据的低位保存在内存低地址中,而数据的高位,保存在内存的高地址中。
为什么会有大端和小端:
我们知道,内存可以看成是一个以字节编址的大数组,以字节为单位,一个字节是8bit。但在C语言中,32位的程序除了8bit的char外,还有16bit的short,32bit的int等等。那么多于8bit,例如16bit,32bit就存在存放顺序的问题,这就导致了大小端存储模式。
大多数Intel兼容机只使用小端模式,IBM和Oracle大多数机器使用大端模式。
当然也存在双端法,可以配置成大端或者小端,不过实际情况是选择了特定的操作系统,大小端便也固定了。
下面我们来做一个练习:
出自百度2015年系统工程师笔试题:
请简述大端字节序和小端字节序的概念,设计一个小程序来判断当前机器的字节序。(10分)
下面给出代码
#include<stdio.h>
#include<stdlib.h>
//代码1
int check_sys()
{
int i = 1;
return (*(char*)&i);
}
int main()
{
int ret = check_sys();
if (ret)
{
printf("小端\n");
}
else
{
printf("大端\n");
}
system("pause");
return 0;
}
//代码2
int check_sys()
{
union
{
int i;
char c;
}un;
un.i = 1;
return un.c;
}
总结
以上就是今天要讲的内容,本文仅仅简单介绍了数据在内存的存储,想了解更多的朋友建议去阅读csapp第二章,好了,感谢大家阅读。