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

初识C语言(数据在内存中的存储) - 实践

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档

文章目录

  • 前言
    • 这篇博客主要围绕整数、大小端字节序、浮点数在内存中的存储。
  • 一、整数在内存中的存储
    • 1. 二进制表示形式
    • 2. 存储方式
    • 3. 常见整数类型
    • 4. 特殊整数处理
    • 5. 内存布局示例
  • 二、⼤⼩端字节序和字节序判断
    • 1. 什么是字节序
    • 2. 为什么需要关注字节序
    • 3. 判断当前系统的字节序
      • 方法1:使用C程序判断
      • 方法2:使用Python判断
      • 方法3:使用命令行工具(Linux)
    • 4. 字节序转换函数
      • C语言中的转换函数
      • Python中的转换
      • 手动实现字节序转换
    • 5. 实际应用示例
      • 网络协议处理
      • 文件格式处理
      • 嵌入式开发
    • 6. 常见问题与解决方案
  • 三、浮点数在内存中的存储
    • 1. IEEE 754标准
    • 2. 存储结构详解
      • 符号位(Sign)
      • 指数部分(Exponent)
      • 尾数部分(Mantissa/Significand)
    • 3. 特殊值表示
    • 4. 舍入规则
    • 5. 精度问题
    • 6. 实际应用注意事项
  • 总结


前言

这篇博客主要围绕整数、大小端字节序、浮点数在内存中的存储。

一、整数在内存中的存储

1. 二进制表示形式

整数在计算机内存中是以二进制形式存储的,使用补码表示法。补码系统有以下特点:

例如,8位整数:

2. 存储方式

整数在内存中的存储遵循以下规则:

  1. 大小端存储

    • 大端模式(Big-endian):高位字节存储在低地址
    • 小端模式(Little-endian):低位字节存储在低地址
    • 例如0x12345678在内存中的存储:
      • 大端:12 34 56 78
      • 小端:78 56 34 12
  2. 对齐方式

    • 现代CPU通常要求数据按照其字长对齐
    • 32位系统通常要求4字节对齐
    • 64位系统通常要求8字节对齐

3. 常见整数类型

不同编程语言中的整数类型及其典型存储大小:

类型C/C++JavaPython存储大小
有符号8位charbyteint1字节
无符号8位unsigned char--1字节
有符号16位shortshortint2字节
无符号16位unsigned shortchar-2字节
有符号32位intintint4字节
无符号32位unsigned int--4字节
有符号64位long longlongint8字节
无符号64位unsigned long long--8字节

注意:Python的int类型会自动扩展以适应大整数,实际存储大小会动态变化。

4. 特殊整数处理

  1. 零的表示

  2. 最小负数的表示

  3. 溢出处理

5. 内存布局示例

32位整数0xAABBCCDD在小端系统中的内存布局:

地址   内容
0x1000 DD
0x1001 CC
0x1002 BB
0x1003 AA

理解整数在内存中的存储方式对于以下场景很重要:

  • 网络协议开发(处理字节序)
  • 二进制文件解析
  • 内存敏感型应用开发
  • 跨平台数据交换

二、⼤⼩端字节序和字节序判断

1. 什么是字节序

字节序(Endianness)指的是多字节数据在计算机内存中的存储顺序。主要有两种类型:

  1. 大端字节序(Big-Endian)

  2. 小端字节序(Little-Endian)

2. 为什么需要关注字节序

字节序问题主要在以下场景中需要注意:

  1. 跨平台数据传输

    • 网络通信(网络协议通常采用大端序)
    • 文件格式交换
    • 不同架构设备间的数据共享
  2. 二进制数据处理

    • 解析网络数据包
    • 读取二进制文件
    • 处理硬件寄存器
  3. 调试和逆向工程

    • 分析内存数据时需要知道字节序
    • 理解寄存器和内存中的值对应关系

3. 判断当前系统的字节序

方法1:使用C程序判断

#include <stdio.h>int main() {unsigned int num = 0x12345678;unsigned char *p = (unsigned char *)&num;if (*p == 0x12) {printf("Big-Endian\n");} else if (*p == 0x78) {printf("Little-Endian\n");} else {printf("Unknown Endianness\n");}return 0;}

方法2:使用Python判断

import sys
if sys.byteorder == 'little':
print("Little-Endian")
else:
print("Big-Endian")

方法3:使用命令行工具(Linux)

lscpu | grep "Endian"
# 或
echo -n I | od -to2 | head -n1 | cut -f2 -d" " | cut -c6
# 输出1表示小端,0表示大端

4. 字节序转换函数

C语言中的转换函数

#include <arpa/inet.h>uint32_t htonl(uint32_t hostlong);  // 主机到网络(32位)uint16_t htons(uint16_t hostshort); // 主机到网络(16位)uint32_t ntohl(uint32_t netlong);   // 网络到主机(32位)uint16_t ntohs(uint16_t netshort);  // 网络到主机(16位)

Python中的转换

import socket
value = 0x12345678
network_order = socket.htonl(value)  # 主机序转网络序
host_order = socket.ntohl(network_order)  # 网络序转主机序

手动实现字节序转换

32位整数的转换示例:

uint32_t swap_endian(uint32_t val) {
return ((val << 24) & 0xff000000) |
((val << 8)  & 0x00ff0000) |
((val >> 8)  & 0x0000ff00) |
((val >> 24) & 0x000000ff);
}

5. 实际应用示例

网络协议处理

处理TCP/IP首部时需要注意:

文件格式处理

常见二进制文件的字节序:

  • PNG:大端序
  • GIF:小端序
  • JPEG:大端序
  • ELF:与目标平台相关

嵌入式开发

访问硬件寄存器时需要特别注意:

6. 常见问题与解决方案

  1. 数据错位问题

    • 现象:接收到的数据与预期不符
    • 解决方案:检查发送方和接收方的字节序是否一致
  2. 跨平台兼容性

    • 使用标准网络字节序(大端)进行数据交换
    • 在文件头中加入字节序标识
  3. 性能考虑

    • 在已知字节序的系统中避免不必要的转换
    • 批量转换优于逐个转换
  4. 调试技巧

    • 使用十六进制查看器检查原始数据
    • 打印内存内容对比预期值

三、浮点数在内存中的存储

1. IEEE 754标准

浮点数在内存中的存储遵循IEEE 754标准,该标准定义了浮点数的二进制表示格式。主要有两种格式:

  • 单精度浮点数(32位):

    • 1位符号位
    • 8位指数位
    • 23位尾数位
  • 双精度浮点数(64位):

    • 1位符号位
    • 11位指数位
    • 52位尾数位

2. 存储结构详解

符号位(Sign)

  • 位于最高位
  • 0表示正数,1表示负数
  • 示例:-3.14的符号位为1

指数部分(Exponent)

尾数部分(Mantissa/Significand)

  • 采用隐含最高位1的表示方法(规范化数)
  • 只存储小数部分
  • 示例:1.1011只需存储1011

3. 特殊值表示

IEEE 754标准还定义了特殊值的表示:

4. 舍入规则

IEEE 754定义了四种舍入模式:

  1. 向最近值舍入(默认)
  2. 向零舍入
  3. 向正无穷舍入
  4. 向负无穷舍入

5. 精度问题

由于浮点数的二进制表示特性,会导致一些十进制小数无法精确表示,如:

6. 实际应用注意事项

在编程中需要注意:

  1. 避免直接比较浮点数相等

    • 应该比较两者差值是否小于某个极小值(如1e-6)
  2. 注意累积误差

    • 多次运算可能导致误差累积
  3. 选择合适的浮点类型

    • 根据精度需求选择单精度或双精度
  4. 注意特殊值的处理

    • 检查NaN和无穷大的情况

总结

以上就是本文总结的内容,关于数据在内存中的存储。

http://www.jsqmd.com/news/257706/

相关文章:

  • 2026年无缝钢管推荐:基于行业合作网络的供应商评价与排名评测 - 十大品牌推荐
  • 海尔洗衣机全球17连冠:从产品到智慧场景的升级
  • 一文搞定 MediaPipe:零基础搭建到核心功能开发的完整教程
  • 2026年无缝钢管推荐:基于行业合作深度评价的供应商排名与综合评测 - 十大品牌推荐
  • 8.3 AI员工性能优化:提升处理效率与准确性
  • 云浮市云城云安新兴郁南罗定区英语雅思培训辅导机构推荐,2026权威出国雅思课程中心学校口碑排行榜 - 老周说教育
  • jumpserver-使用
  • SGMICRO圣邦微 SGM8274XS14G/TR SOP14 运算放大器
  • 2026年产品设计公司选购看什么?这份深度评测与口碑排名推荐给你答案 - 十大品牌推荐
  • Redis性能优化:高效查找1亿Key中的10万目标技巧
  • 2026年外观设计公司选购应关注什么?这份深度评价与推荐排名提供答案 - 十大品牌推荐
  • 2026年外观设计公司推荐:聚焦口碑对比的深度评价及最终排名解析 - 十大品牌推荐
  • 2026亲测!有机肥好厂口碑推荐分享
  • 2026年产品设计公司推荐:基于权威机构评定的TOP5排名与深度评测 - 十大品牌推荐
  • [C++][cmake]基于C++在windows上onnxruntime+opencv部署yolo26的图像分类onnx模型
  • 吐血推荐10个AI论文写作软件,本科生毕业论文轻松搞定!
  • 如何选择产品设计合作伙伴?2026年最新深度评测与综合排名推荐 - 十大品牌推荐
  • 面向边缘设备的YOLOv12模型轻量化:剪枝与知识蒸馏联合优化方法
  • 收集自己的每日消费金额,统计每周消费峰值,输出消费高峰日,并给出省钱建议。
  • YOLOv13 增强指南:集成空间注意力(SAB)的RAB卷积模块实现与性能验证
  • 基于SpringBoot+Vue的电商应用系统(11718)
  • 攻克海思芯片部署难点:YOLOv8模型RPN硬化与INT8量化的精度保持方案
  • 直播电商监督管理办法
  • 基于SpringBoot的失物招领平台的设计与实现(11719)
  • SpringBoot集成Hera,分布式应用监控与追踪解决方案
  • 网络暴力信息治理规定
  • 亲测好用!专科生毕业论文必备9大AI论文网站测评
  • 2026年外观设计公司推荐:2026年度横向对比评测与用户评价排名报告 - 十大品牌推荐
  • 低代码高频实践场景系列之五——跨境零售PLM系统 - 实践
  • 网络和安全防护建设方案(Word)