数据类型
char 1字节 unsigned char signed char//下面三个默认定义为有符号类型,char类型取决于编译器
short 2字节 unsigned short signed short
int 4字节 unsigned int signed int
long 4/8字节(取决于编译器) unsigned long signed long
float 4字节
double 8字节
long double 自定义类型
1字节=8bit
首先我们要知道数据在所开辟内存中以补码的形式存储
1.那么什么是补码
把数据转换成其二进制形式;(平时我们写的数都是十进制)
1.正数的符号位为0,负数符号位为1
例如int a=1;int类型4个字节32bit位,正数的原码,反码,补码,相同
原码00000000000000000000000000000001
反码00000000000000000000000000000001
补码00000000000000000000000000000001
2.负数的原码,除了符号位按位取反,就是其反码,反码+1就是其补码
int a=-1;
原码10000000000000000000000000000001
反码1111111111111111111111111111111111110
补码1111111111111111111111111111111111111
3.简单调试技巧
我们可以自己在VS里定义一个int a=10;通过调试来观察其在内存中的存储方式,定义完成后按F10开始调试;点击调试,窗口,内存,然后随意选择一个内存1即可观察
10的补码为00000000000000000000000000001010,开辟四个字节空间,存进是4个数一组,转换为16进制,0000 1010,1*2^3+1*2^1=10,对应十六进制中的a,0a
十六进制0,1,2,3,4,5,6,7,8,9,a,b,c,d,e,f
不要忘了按F10,或F11不然a还没赋值,观察到的就不是0a,而是其他的数字,如下图
F10为逐语句调试,F11为逐过程调试,简单来说,如果有个调用函数,F11会进入到函数过程中,而F10不进入这个函数中,直接跳过去。
这里还涉及到大小端存储模式,简单来说数据开辟内存从高地址到低地址,数据存储时数据低位放到高地址处为大端字节序存储,反之为小端字节序存储,列如上述的10,转换后为00 00 00 0a。存储时0a放到了低地址0x001FFC14处,为小端字节序存储模式。
下面来一段简单代码练习一下
#include<stdio.h>
int main()
{
char a=-1;
signed char b=-1;
unsigned char c=-1;
printf("a=%d,b=%d,c=%u",a,b,c);
return 0;
}
这里的unsigned为无符号数,意思就是上面所介绍的符号位没有了正负的效果,全部识别为正数,列如1000000000000000000000000000001,不再是-1,而是
-1的补码11111111111111111111111111111111,把一个整型数据存入到char,相当于把大象塞进冰箱,硬塞肯定是不行的,我们要把数据截断进行存储,从后往前截取8个bit位,11111111。
%u按照无符号整型打印,%d按有符号整型打印,代码中的a存入11111111,按%d打印,这个时候我们需要进行整型提升
整型提升
有符号数,提升补符号位,无符号数提升补0,提升的时候是看定义时的类型来提升,char在VS编译中默认有符号类型,所以提升补1(按这个11111111补)
浮点数存储简要介绍
float.h中定义了浮点数的取值范围相关信息,而limits.h定义了整数的。
例子 1.5
写成二进制形式1.101(错的)
浮点数存储形式..........2^0 小数点后存储形式 2^(-1) 2^(-2) 2^(-3).....
1/2=0.5,所以实际上1.5的二进制形式为1.1
二进制浮点数可以写成此形式(-1)^S*M*2^E,
S=0时该数为正数,S=1时为负数
M表示有效数字,2>M≥1,
1.5的M就是1.1,
E为小数点前移或后移的位数,例如5.5二进制形式101.1,1.011,E为2
计算机存储时存的是S,M,E
计算机保存M时,默认这个数的第一位总是1,会被舍去,只保存小数点后面的数字,等到读取的时候再加上去。
E的存储,E为一个无符号整数,由于规定正整数部分为1,E会出现负数,例如0.5,二进制形式0.1,需将小数点右移动1位,E为-1,对于8位的E需要先加127再存入,对于11位的需要现加1023再存入。
存后没存满后面补0;
32位浮点数存储;
E为8位,取值范围0~255;
64位浮点数存储
E为8位,取值范围0~2047;
浮点数取出(只做简单介绍)
分为3种情况
1.E不全为0或不全为1
E的计算值减去127或1023,得到真实值,再将M前加上第一位的1
2.E全为0
E取出同上,但是有效数字M不再加上第一位的1,而是还原为0.XXXXXXXX的小数,这样做是为了表示±0,以及接近于0的很小的数字
3.E全为1
这时如果有效数字全为0,表示±无穷大。