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

C语言内存函数和数据在内存的存储

一、内存操作函数深度解析

函数名原型核心特性典型应用场景注意事项
memcpyvoid* memcpy(void* dest, const void* src, size_t num)内存块无重叠复制,性能高数组拷贝、结构体复制1. 必须确保目标空间足够
2. 不支持重叠内存(用memmove替代)
3. 按字节操作,无类型检查
memmovevoid* memmove(void* dest, const void* src, size_t num)支持重叠内存复制,自动调整复制方向缓冲区自我覆盖操作(如字符串移位)性能略低于memcpy(需判断内存关系)
memsetvoid* memset(void* ptr, int value, size_t num)按字节填充内存初始化内存块(清零、设置标记位)1. value取值范围为0x00~0xFF
2. 不可用于初始化非字节类型数组(如int[]需循环赋值)
memcmpint memcmp(const void* ptr1, const void* ptr2, size_t num)按字节精确比较内存内容二进制数据校验、结构体比对strcmp不同,不会因\0终止比较
代码示例:memcpymemmove对比

#include <stdio.h>
#include <string.h>

int main() {
    char str[] = "ABCDEFGH";
    
    // 使用memcpy(未定义行为,源和目标重叠)
    memcpy(str + 2, str, 5);  
    printf("memcpy结果: %s\n", str); // 输出可能异常(如ABABABA)

    // 使用memmove(安全复制)
    strcpy(str, "ABCDEFGH");         // 重置字符串
    memmove(str + 2, str, 5);       
    printf("memmove结果: %s\n", str); // 正确输出ABABCDE
    return 0;
}

二、数据存储底层机制

1. 整数存储模型
编码方式
码制正整数表示负整数表示(以-5为例)特点
原码直接二进制1+绝对值二进制(1 0000101直观但加减运算复杂
反码同原码符号位不变,其他位取反(1 1111010解决减法问题,但存在+0-0
补码同原码反码+1(1 1111011统一加减法,消除-0,现代计算机统一采用
大小端字节序
  • 大端模式 (Big-Endian):数据高位字节存于低地址
    示例0x12345678 存储顺序为 12 34 56 78(网络传输、Java虚拟机)

  • 小端模式 (Little-Endian):数据低位字节存于低地址
    示例0x12345678 存储顺序为 78 56 34 12(x86/ARM架构)

检测代码:

#include <stdio.h>

int check_endian() {
    int num = 0x1;
    return *(char*)&num == 1; // 返回1为小端,0为大端
}
2. 字符型存储
类型存储范围二进制特征类型转换规则
char-128 ~ 127最高位为符号位直接按ASCII码转换('A'65
unsigned char0 ~ 255无符号位,全数据位可用于存储原始字节数据(如图像像素)

陷阱案例:

char c = 200;   // 溢出!实际值为-56(补码)
unsigned char uc = 200; // 正确存储
printf("%d vs %d", c, uc); // 输出-56 vs 200

三、综合应用与调试技巧

  1. 内存操作安全

  • 使用memcpy前需验证目标空间大小:

#define SAFE_COPY(dest, src, n) do { \
    assert((dest) != NULL && (src) != NULL); \
    assert((n) <= sizeof(*(dest))); \
    memcpy((dest), (src), (n)); \
} while(0)

  • 优先选择memmove处理不确定内存关系的场景。

2.类型转换与指针操作

  • 正确访问多字节数据:

    int num = 0x12345678;
    unsigned char* p = (unsigned char*)#
    // 输出字节内容(依赖字节序)
    for(int i=0; i<4; i++) 
        printf("%02X ", p[i]); 

    3.调试内存问题

    • 使用工具检测越界访问:

      • Valgrind:检测内存泄漏、越界访问。

      • AddressSanitizer(GCC/Clang):编译时加入-fsanitize=address


四、总结:内存操作的核心原则

  1. 明确数据边界:始终确保操作的内存区域在合法范围内。

  2. 理解底层存储:掌握补码与字节序机制,避免跨平台数据传输错误。

  3. 选择合适工具:根据场景选用memcpy/memmove,区分memset与循环初始化的适用场景。

相关文章:

  • 第一章:Pandas 基础
  • 解决Ubuntu系统鼠标不流畅的问题
  • AI 编程器:Cursor Trae
  • Spring Boot内嵌服务器全解析:Tomcat vs Jetty vs Undertow 选型指南
  • 【Python】案例:计算股票收益率和波动率
  • 深入解析Spring Boot自动装配:原理、设计与最佳实践
  • 常见接口协议介绍
  • 秒杀系统的性能优化
  • Pytorch torch.nn.utils.rnn.pad_sequence 介绍
  • 对访问者模式的理解
  • 压力容器的断裂力学计算
  • ansible+docker+docker-compose快速部署4节点高可用minio集群
  • 2140 星期计算
  • 仿modou库one thread one loop式并发服务器
  • 浅谈进程的就绪状态与挂起状态
  • 【网络协议】WebSocket讲解
  • Kettle如何与应用集成
  • Python星球日记 - 第11天:文件操作
  • 【项目日记】高并发服务器项目总结
  • [环境配置] 1. 开发环境搭建
  • wordpress论坛系统/微信搜索seo优化
  • python官方网站/北京全网推广
  • 政府扶持办厂项目/网站排名优化外包公司
  • 什么网站可以做直播/建网站找谁
  • 信息网站建设预算/资源猫
  • 平乡县网站建设平台/淘宝seo具体优化方法