C语言浮点数存储方式

对于浮点类型的数据采用单精度类型(float)和双精度类型(double)来存储,float数据占用 32bit,double数据占用 64bit.其实不论是float类型还是double类型,在计算机内存中的存储方式都是遵从IEEE的规范的,float 遵从的是IEEE R32.24 ,而double 遵从的是R64.53。   

无论是单精度还是双精度,在内存存储中都分为3个部分:  

1) 符号位(Sign):0代表正,1代表为负;  

2) 指数位(Exponent):用于存储科学计数法中的指数数据,并且采用移位存储; 

3) 尾数部分(Mantissa):尾数部分


其中float的存储方式如下图所示:


而双精度的存储方式为:


R32.24和R64.53的存储方式都是用科学计数法来存储数据的

用二进制的科学计数法第一位都是1嘛,干嘛还要表示呀?可以将小数点前面的1省略,所以23bit的尾数部分,可以表示的精度却变成了 24bit,道理就是在这里。

那24bit能精确到小数点后几位呢,我们知道9的二进制表示为1001,所以4bit能精确十进制中的1位小数 点,24bit就能使float能精确到小数点后6位,而对于指数部分,因为指数可正可负,8位的指数位能表示的指数范围就应该为:-127-128了, 所以指数部分的存储采用移位存储,存储的数据为元数据+127。

下面就看看8.25和120.5在内存中真正的存储方式: 

首先看下8.25,用二进制的科学计数法表示为
:1.0001*2^3  按照上面的存储方式,符号位为0,表示为正;指数位为3+127=130,位数部分为 1.00001,故8.25的存储方式如下:  0xbffff380:    01000001000001000000000000000000 

分解如下:0--10000010--00001000000000000000000  

符号位为0,指数部分为10000010,位数部分为 00001000000000000000000 


同理,120.5在内存中的存储格式如下:  0xbffff384:    01000010111100010000000000000000 

分解如下:0--10000101--11100010000000000000000  


那么如果给出内存中一段数据,并且告诉你是单精度存储的话,你如何知道该数据的十进制数值 呢?其实就是对上面的反推过程,比如给出如下内存数据: 01000001001000100000000000000000 

第一步:符号位为0,表示是正数;  第二步:指数位为10000010,换算成十进制为130,所以指数为130-127=3; 第三步:尾数位为01000100000000000000000,换算成十进制为 (1+1/4+1/64); 所以相应的十进制数值为:2^3*(1+1/4+1/64)=8+2+1/8=10.125

### C语言浮点数存储方式 #### IEEE 754标准下的浮点数结构 在C语言中,`float` 类型的数据遵循IEEE 754标准进行存储。该标准规定了单精度(32位)和双精度(64位)两种主要形式。对于`float`类型而言,采用的是单精度格式。 具体来说,一个32位的`float`被划分为三个部分: - **符号位 (S)**:最高有效位用于表示正负号,0代表正值,1代表负值。 - **指数偏移量 (E, Exponent Bias)**:接下来的8位用来编码实际指数加上一个固定的偏置值(通常为127)。这使得可以表达非常大或非常小的数量级变化。 - **尾数/分数 (M/Fraction or Mantissa)**:剩下的23位保存着规范化后的尾数部分,它实际上是从隐含的最左边‘1’之后开始记录的有效数字序列[^1]。 因此,在这种布局下,任何给定的`float`数值都可以按照如下公式解析出来: \[ (-1)^{\text{sign}} \times 2^{(\text{exponent}-\text{bias})} \times 1.\text{mantissa}\] 其中, - `sign` 是符号位; - `exponent` 表示无符号整数形式的实际指数字段; - `bias` 则是指定的标准偏差值(对于`float`是127),而 - `mantissa` 就是由剩余比特组成的二进制小数串[^2]。 #### 实际案例分析——`7.75` 考虑具体的例子`7.75f`,这是一个典型的十进制实数转换成二进制的过程。首先将其分解为二进制字符串并调整到标准化的形式: \[ 7.75_{(10)} = 111.11_2 = 1.1111\times2^2 \] 这里我们看到基数变成了\(2\)次幂乘以一个小于\(2\)但大于等于\(1\)的小数因子。接着根据上述规则设置各个域的内容: - 符号位设为`0`因为它是正数; - 指数应为\(2+127=129\)即`10000001`; - 尾数去掉前导'1.'后剩下`1111...`(共23位填充零)。 最终得到完整的机器码表示为: ```c union { float f; unsigned int i; }; // 假设变量名为data,则可以通过联合体查看内部字节序 unsigned char bytes[sizeof(float)]; memcpy(bytes, &data.f, sizeof(float)); for(int j=0; j<sizeof(float); ++j){ printf("%02X ", bytes[j]); } ``` 这段代码展示了如何通过联合体访问同一块内存的不同解释视图,并打印出对应的十六进制字节数组来观察`7.75f`的具体存储形态[^4]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值