目录
🥑🍓Let's go🍓🥑
一:数据类型介绍
char // 字符数据类型short // 短整型int // 整形long // 长整型long long // 更长的整形float // 单精度浮点数double // 双精度浮点数
类型的意义:
1.使用这个类型开辟内存空间的大小(大小决定了适用范围)
2.如何看待内存空间的视角。
1.1类型的基本归类
1.1-1整型家族
charunsigned charsigned charshortunsigned short [ int ]signed short [ int ]intunsigned intsigned intlongunsigned 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浮点数家族
floatdouble
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
);
🌈🌈文章结束☀️☀️
这份博客👍如果对你有帮助,给博主一个免费的点赞以示鼓励欢迎各位🔎点赞👍评论收藏⭐️,谢谢!!!
如果有什么疑问或不同的见解,欢迎评论区留言欧👀
如果有什么疑问或不同的见解,欢迎评论区留言欧👀