进阶C语言-数据在内存中的存储

目录

​​​​​​一:数据类型介绍

        1.1类型的基本归类

        1.1-1整型家族

        1.1-2浮点数家族

        1.1-3构造类型

        1.1-4指针类型

        1.1-5空类型

二:整型在内存中的存储

        2.1原码、反码、补码

        2.2大小端介绍

                2.2-1 大小端介绍

                2.2-2 判断编辑器为大端存储还是小端存储(4种方式)

        2.3练习

                 2.3-1 练习一

        2.3-2 练习二

        2.3-3 练习三

        2.3-4 练习四

        2.3-5 练习五

        2.3-6 练习六

三:浮点型在内存中的存储

        3.1 一个例子

        3.2 浮点数存储规则 


🥑🍓Let's go🍓🥑

一:数据类型介绍

char                // 字符数据类型
short              // 短整型
int                 // 整形
long             // 长整型
long long   // 更长的整形
float          // 单精度浮点数
double     // 双精度浮点数
类型的意义:
1.使用这个类型开辟内存空间的大小(大小决定了适用范围)
2.如何看待内存空间的视角。

        1.1类型的基本归类

1.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 ]
【注意】: 1.int=signed int ;short=signed; long=signed long;
                  2.易错之处:char!=(不一定相等)signed char(这个要取决于编译器)
                  3.

                        n位无符号整数,取值范围:

                        0    ~     ( 2^n - 1 )

                        n位有符号整数,取值范围:

                        - (2^(n-1))     ~      + (2^(n-1) - 1)

1.1-2浮点数家族

float
double

1.1-3构造类型

数组类型
结构体类型 struct
枚举类型 enum
联合类型 union
【注意】: 例如:int arr[5]的类型是int [5];

1.1-4指针类型

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

1.1-5空类型

void 表示空类型(无类型)

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

二:整型在内存中的存储

        2.1原码、反码、补码

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

        原码:直接将数值按照正负数的形式翻译成二进制就可以得到原码。

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

        补码:反码+1就得到补码。

【注意】:对于整型来说:数据存放内存中其实存放的是补码。

🌈🌈例如:(一些整型的相关打印)

1:
#include <stdio.h>
int main()
{
	char a = -1;
	//原码:10000000000000000000000000000001
	//反码:11111111111111111111111111111110
	//补码:11111111111111111111111111111111
	//因为char类型是一个字节,只有8bite,则从右往左取8位
	//得:11111111
	//整型提升得:
	//11111111111111111111111111111111(补码)
	//11111111111111111111111111111110(反码)
	//10000000000000000000000000000001(原码)  -1
	//当在打印得时候显示的是原码
	signed  char b = -1;
	//此时b=-1;
	unsigned char c = -1;
	//原码:10000000000000000000000000000001
	//反码:11111111111111111111111111111110
	//补码:11111111111111111111111111111111
	//无符号char在整型提升时前面补0
	//00000000000000000000000011111111
	//无符号整型的原码、反码和补码相同
	//由原码得c=255
	printf("a=%d,b=%d,c=%d", a, b, c);
	//结果为a=-1,b=-1,c=255
	return 0;
}

2:#include<stdio.h>
int main()
{
	char a = -128;
	//原码:10000000000000000000000010000000
	//反码:11111111111111111111111101111111
	//补码:11111111111111111111111110000000
	//截取:10000000
	//整型提升得
	//11111111111111111111111110000000(补码)
	//打印的结果为4294967168
	printf("%u", a);
	return 0;
}
【注意】:
这个打印不管他是什么变量,都是以无符号数的形式去打印内存中的内容,而内存中存的是负数的补码

3:
#include <stdio.h>
int main()
{
	char a = 128;
	//原码:00000000000000000000000010000000
	//整形提升:
//11111111111111111111111110000000(补码)
//打印的结果为4294967168
	printf("%u ", a);
	return 0;
}

        2.2大小端介绍

2.2-1 大小端介绍

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

2.2-2 判断编辑器为大端存储还是小端存储(4种方式)

1:
#include <stdio.h>
int main()
{
	int a =1;
	char* pa = (char*)&a;
	if (*pa == 1)
	{
		printf("小端\n");
	}
	else
	{
		printf("大端\n");
	}
	return 0;
}

2:
#include <stdio.h>
int cheak_sys()
{
	int a = 1;
	char* pa = (char*)&a;
	if (*pa == 1)
		return 1;
	else
		return 0;
}
int main()
{
	if (cheak_sys() == 1)
	{
		printf("小端\n");
	}
	else
	{
		printf("大端\n");
	}
}

3:
#include <stdio.h>
int cheak_sys()
{
	int a = 1;
	char* pa = (char*)&a;
	//也可以使用三目操作符;
	return *pa == 1 ? 1 : 0;
}
int main()
{
	if (cheak_sys() == 1)
	{
		printf("小端\n");
	}
	else
	{
		printf("大端\n");
	}
}

4:
#include <stdio.h>
int cheak_sys()
{
	int a = 1;
	if (*(char*)&a == 1)
	{
		return 1;
	}
	else
	{
		return 0;
	}
}
int main()
{
	if (cheak_sys() == 1)
	{
		printf("小端\n");
	}
	else
	{
		printf("大端\n");
	}
	return 0;
}

        2.3练习

🌈🌈2.3-1 练习一

#include <stdio.h>
int main()
{
	char a = -1;
    signed char b = -1;
	unsigned char c = -1;
	//原码:10000000 00000000 00000000 00000001
	//反码:11111111111111111111111111111111110
	//补码:11111111 11111111 11111111 11111111
	//整型提升:00000000 00000000 00000000 11111111
	//c的值为255
	printf("a=%d,b=%d,c=%d", a, b, c);
	return 0;
//结果为: a=-1,b=-1,c=255;
}

🌈🌈2.3-2 练习二

#include <stdio.h>
int main()
{
	char a = -128;
	//原码:10000000 00000000 00000000 10000000
	//反码:11111111 11111111 11111111 01111111
	//补码:11111111 11111111 11111111 10000000
	printf("%u\n", a);
	//a=4294967168
	//以%u的形式打印时就是打印该数在内存中的补码
	return 0;
}

🌈🌈2.3-3 练习三

#include <stdio.h>
int main()
{
	char a = 128;
 //原码:00000000 00000000 00000000 10000000
 //整型提升:
//11111111 11111111 11111111 10000000
	printf("%u\n", a);
a=4294967168
	return 0;
}

🌈🌈2.3-4 练习四

#include <stdio.h>
int main()
{
	int i = -20;
	//原码:10000000 00000000 00000000 00010100
	//反码:11111111 11111111 11111111 11101011
	//补码:11111111 11111111 11111111 11101100
	unsigned int j = 10;
	//原码: 00000000 00000000 00000000 00001010
	//i+j:
	//11111111 11111111 11111111 11101100
	//00000000 00000000 00000000 00001010
	//11111111 11111111 11111111 11110110(补码)
	//11111111 11111111 11111111 11110101(反码)
	//10000000 00000000 00000000 00001010(原码)
	printf("%d\n", i + j);
	i+j的结果为-10
	//输出是以原码的形式输出
	return 0;
}

🌈🌈2.3-5 练习五

#include<stdio.h>
int main()
{
	unsigned int i;
	for (i = 9; i >= 0; i--)
	{
		printf("%u\n", i);
	}
	//-1的原码:10000000 00000000 00000000 00000001
	//-1的反码:11111111 11111111 11111111 11111110
	//-1的补码:11111111 11111111 11111111 11111111
    //-1在内存中的值为4,294,967,295
	return 0;
所以该程序会死循环
}

🌈🌈2.3-6 练习六

#include <stdio.h>
int main()
{
	char a[1000];
	int i;
	for (i = 0; i < 1000; i++)
	{
		a[i] = -1 - i;
	}
	printf("%d", strlen(a));
	return 0;
}
strlen求的时‘\0’之前的长度。
(127-0)~(-1--128)
127+128=255

三:浮点型在内存中的存储

        3.1 一个例子

#include <stdio.h>
int main()
{
	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);
	return 0;
}

打印结果为:

        3.2 浮点数存储规则 

任意一个二进制浮点数v可以表示成以下的形式

(-1)^S*M*2^E

(-1)*S表示符号位;当S=0,V为正数;S=1,V为负数。

M表达有效数字,大于等于1,小于2。

2^E表示指数位

【例如】

十进制的 5.0 ,写成二进制是 101.0 ,相当于 1.01×2^2
那么,按照上面 V 的格式,可以得出 S=0 M=1.01 E=2
十进制的 -5.0 ,写成二进制是 - 101.0 ,相当于 - 1.01×2^2 。那么, S=1 M=1.01 E=2

【注意】:

1:首先,E为一个无符号整数(unsigned int

这意味着,如果 E 8 位,它的取值范围为 0~255 ;如果 E 11 位,它的取值范围为 0~2047。但是,我们知道,科学计数法中的 E是可以出现负数的,所以 IEEE 754规定,存是 127 ;对于 11 位的 E 这个中间数是1023。比如, 2^10 E 10 ,所以保存成 32位浮点数时,必须保存成 10+127=137
即10001001
2:然后,指数 E从内存中取出还可以再分成三种情况:
1️⃣ E不全为 0 或不全为 1
0.5 1/2 )的二进制形式为 0.1 ,由于规定正数部分必须为 1 ,即将小数点右移 1 位,则为
1.0*2^(-1) ,其阶码为 -1+127=126 ,表示为
01111110 ,而尾数 1.0 去掉整数部分为 0 ,补齐 0 23 00000000000000000000000 ,则其二进
制表示形式为: 0 01111110 00000000000000000000000
2️⃣ E全为 0
这时,浮点数的指数 E 等于 1-127 (或者 1-1023 )即为真实值,
有效数字 M 不再加上第一位的 1 ,而是还原为 0.xxxxxx 的小数。这样做是为了表示 ±0 ,以及接近于
0 的很小的数字。
3️⃣ E全为 1
这时,如果有效数字 M 全为 0 ,表示 ± 无穷大(正负取决于符号位 s );
    🌈🌈文章结束☀️☀️
这份博客👍如果对你有帮助,给博主一个免费的点赞以示鼓励欢迎各位🔎点赞👍评论收藏⭐️,谢谢!!!
如果有什么疑问或不同的见解,欢迎评论区留言欧👀
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值