当前位置: 首页 > article >正文

c语言:整数与浮点数在内存中的存储方式

整数在内存中的存储:

在计算机内存中,整数通常以二进制形式存储。计算机使用一定数量的比特(bit)来表示整数,比如32位或64位。在存储整数时,计算机使用补码形式来表示负数,而使用原码形式来表示正数。整数的存储方式使得计算机能够进行加减乘除等运算。


浮点数在内存中的存储:

与整数不同,浮点数采用IEEE 754标准来存储。在内存中,浮点数由三个部分组成:符号位、指数位和尾数位。这种存储方式使得计算机能够表示小数和非常大或非常小的数字,但也带来了精度丢失的问题。


IEEE 754标准

IEEE 754标准定义了浮点数的存储格式,包括单精度浮点数(32位)和双精度浮点数(64位)。在这两种格式中,浮点数的存储结构都由三个部分组成:符号位、指数位和尾数位。

  1. 符号位:用于表示浮点数的正负,0表示正数,1表示负数。

  2. 指数位:用于表示浮点数的阶码(指数部分),指数位的长度决定了浮点数的表示范围。

  3. 尾数位:用于表示浮点数的尾数,尾数位的长度决定了浮点数的精度。

单精度浮点数(32位)

单精度浮点数采用32位来存储,其中1位用于符号位,8位用于指数位,23位用于尾数位。

符号位(1位) 指数位(8位) 尾数位(23位)

双精度浮点数(64位)

双精度浮点数采用64位来存储,其中1位用于符号位,11位用于指数位,52位用于尾数位。

符号位(1位) 指数位(11位) 尾数位(52位)

 

特别注意: 

当我们使用单精度浮点数(32位)来存储一个小数时,可能会出现精度丢失的情况。例如,让我们考虑存储十进制数 0.1 在单精度浮点数中的表示。

在十进制中,0.1 可以用有限位数准确表示,但在二进制中,0.1 是一个无限循环小数。因此,在单精度浮点数中,0.1 的精确表示是不可能的。

让我们看看单精度浮点数中存储 0.1 的近似表示:

首先,将 0.1 转换为二进制。0.1 的二进制表示是 0.00011001100110011001100110...,这是一个无限循环。

然后,根据单精度浮点数的格式,我们将其表示为科学计数法形式:1.100110011001100110011002 x 2^-4。

在单精度浮点数中,我们需要舍入到23位尾数。因此,最终存储的近似值是:

符号位:0(正数)
指数位:-4 + 127(偏移值,127是单精度浮点数的偏移量)
尾数位:10011001100110011001100

这个存储的近似值会导致精度丢失,因为实际的0.1在单精度浮点数中无法被精确表示。在进行浮点数运算时,这种精度丢失可能会积累导致不准确的结果。

举个例子:

#include <stdio.h>
#include <stdint.h>

int main() 
{
	float num = 0.1f; // 使用单精度浮点数表示0.1
	// 将浮点数的内存表示以十六进制打印出来
	uint32_t* ptr = (uint32_t*)&num;
	printf("0.1的内存表示为: 0x%08x\n", *ptr);

	return 0;
}

7faa5db4797b431e951d9d392bf81caf.png

 

#include <stdio.h>
#include <stdint.h>

int main() 
{
    float num = -27.25f; // 使用单精度浮点数表示-27.25

    // 将浮点数的内存表示以十六进制打印出来
    uint32_t* ptr = (uint32_t*)&num;
    printf("内存中的表示为: 0x%08x\n", *ptr);

    // 分别提取符号位、指数位和尾数位
    uint32_t sign = (*ptr >> 31) & 0x1;
    uint32_t exponent = (*ptr >> 23) & 0xFF;
    uint32_t mantissa = *ptr & 0x7FFFFF;

    printf("符号位: %d\n", sign);
    printf("指数位: 0x%x\n", exponent);
    printf("尾数位: 0x%x\n", mantissa);

    return 0;
}

63dff24654c942b498cdf9c017c1b914.png

符号位:1位
┌─────┬───────────────────────┬─────────────────────────────────────┐
|  S  |       指数位(E)      |              尾数位(M)              |
└─────┴───────────────────────┴─────────────────────────────────────┘
  31                       23                                  0

例如  1.5,

符号位:1位
┌─────┬───────────────────────┬─────────────────────────────────────┐
|  0  |       01111111        |       10000000000000000000000       |
└─────┴───────────────────────┴─────────────────────────────────────┘
  31                       23                                  0
  • 符号位(S):用于表示浮点数的正负,0表示正数,1表示负数。
  • 指数位(E):用于表示浮点数的指数部分,通过偏移(bias)来表示实际的指数值。
  • 尾数位(M):用于表示浮点数的尾数部分,包括小数点后面的数字。

 

9c390aaba26d4f8bb4f15b0a48dc4430.png

 


http://www.kler.cn/a/154605.html

相关文章:

  • 【Apache Paimon】-- 1 -- Apache Paimon 是什么?
  • stm32——通用定时器时钟知识点
  • 部署Apache Doris
  • 什么是项目完整性管理?
  • HarmonyOS Next 组件或页面之间的所有通信(传参)方法总结
  • 释放高级功能:Nexusflows Athene-V2-Agent在工具使用和代理用例方面超越 GPT-4o
  • 抖音视频如何无水印保存?抖音视频无水印保存教程
  • k8s部署jenkins
  • 【前端】JS实现SQL格式化
  • Adobe Indesign操作
  • 『测试基础』| 如何理解测试用例管理和缺陷管理?
  • 【海思SS528 | VO】MPP媒体处理软件V5.0 | 视频输出模块——学习笔记
  • RxJava
  • Halcon tiff 点云读取以及平面矫正
  • 算法工程师面试八股(搜广推方向)
  • jenkins使用nexus插件
  • 如何创建一个vue工程
  • CSS浅谈动画性能
  • 密码学学习笔记(二十二):RSA签名方案
  • CC2530basic_Rf串口无线收发
  • [蓝桥杯 2020 省 AB1] 解码
  • 已解决AttributeError: module ‘gradio‘ has no attribute ‘outputs‘
  • Java集合类的重要性
  • 误用STM32串口发送标志位 “USART_FLAG_TXE” “USART_FLAG_TC”造成的BUG
  • GPT实战系列-大模型训练和预测,如何加速、降低显存
  • 【涨薪技术】深入接口测试之Mock技术