c语言内存函数的深度解析

简介: c语言内存函数的深度解析

本章对 memcpy,memmove,memcmp 三个函数进行详解和模拟实现;

本章重点:3个常见内存函数的使用方法及注意事项并学会模拟实现;

如果您觉得文章不错,期待你的一键三连哦,你的鼓励是我创作的动力之源,让我们一起加油,一起奔跑,让我们顶峰相见!!!

1.memcpy函数(内存拷贝函数)

函数介绍

1.memcpy函数的作用:从source的位置开始向后复制num个字节的数据到destination的内存位置。 可以拷贝字符数组,整型数组,结构体等等,所以参数是void* 的指针;

2.这个函数在遇到 '\0' 的时候并不会停下来。(注意)

3.memcpy函数的局限:如果source和destination有任何的重叠,复制的结果都是未定义的。

memcpy函数拷贝不重叠的内存的拷贝。

4.拷贝结束,返回目标空间的起始地址。

使用举例:

memcpy的模拟实现:

void* memcpy(void* dst, const void* src, size_t count)
{
  void* ret = dst;
  assert(dst);
  assert(src);
  while (count--) 
    {
    *(char*)dst = *(char*)src;
    dst = (char*)dst + 1;
    src = (char*)src + 1;
  }
  return(ret);
}

就是将源内容一个一个字节的拷贝到目标空间去,所以强制类型转换成char*;

2.memmove函数(内存移动函数)

函数介绍

memcpy和memmove函数的用法一样;作用有所差别;

1.memmove函数不仅可以拷贝不重叠的内存,而且可以拷贝重叠的内存。

2.和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。

3.如果源空间和目标空间出现重叠,就得使用memmove函数处理。

memmove的使用示例

当memmove函数用来拷贝重叠内存

memmove函数的模拟实现

void* memmove(void* dst, const void* src, size_t count)
{
  void* ret = dst;
  if (dst <= src || (char*)dst >= ((char*)src + count))
  {
    while (count--)
    {
      *(char*)dst = *(char*)src;
      dst = (char*)dst + 1;
      src = (char*)src + 1;
    }
  }
  else 
  {
    dst = (char*)dst + count - 1;
    src = (char*)src + count - 1;
    while (count--) 
    {
      *(char*)dst = *(char*)src;
      dst = (char*)dst - 1;
      src = (char*)src - 1;
    }
  }
  return(ret);
}

解析:

例如

将arr[]={1,2,3,4,5,6,7,8,9,10}中的1 2 3 4 5 放到3 4 5 6 7的位置上去,如果我们还是用前面memcpy的思想

第一步:把1放到3上去,变为1 2 1 4 5 6 7 8 9 10

第二步:把2放到4上去,变为1 2 1 2 5 6 7 8 9 10

第三步:我们需要将3放到5上去,但是,第一步我们已经将3改变了,所以这种思想不行;

memmove模拟实现思想:

 

3.memcmp函数(内存比较函数)

函数介绍

比较从ptr1和ptr2指针开始的num个字节

返回值如下:

使用示例代码:

int main()
{
  char buffer1[] = "DWgaOtP12df0";
  char buffer2[] = "DWGAOTP12DF0";
  int n;
  n = memcmp(buffer1, buffer2, sizeof(buffer1));
  if (n > 0)
  {
    printf("'%s' is greater than '%s'.\n", buffer1, buffer2);
  }
  else if (n < 0)
  {
    printf("'%s' is less than '%s'.\n", buffer1, buffer2);
  }
  else
  {
    printf("'%s' is the same as '%s'.\n", buffer1, buffer2);
  }
  return 0;
}

运行结果:

本章完~


目录
相关文章
|
2月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
126 26
|
2月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
241 15
|
9月前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
695 14
|
9月前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
332 6
|
9月前
|
存储 算法 C语言
【C语言】深入浅出:C语言链表的全面解析
链表是一种重要的基础数据结构,适用于频繁的插入和删除操作。通过本篇详细讲解了单链表、双向链表和循环链表的概念和实现,以及各类常用操作的示例代码。掌握链表的使用对于理解更复杂的数据结构和算法具有重要意义。
2908 6
|
9月前
|
存储 网络协议 算法
【C语言】进制转换无难事:二进制、十进制、八进制与十六进制的全解析与实例
进制转换是计算机编程中常见的操作。在C语言中,了解如何在不同进制之间转换数据对于处理和显示数据非常重要。本文将详细介绍如何在二进制、十进制、八进制和十六进制之间进行转换。
825 5
|
9月前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
194 5
|
9月前
|
存储 算法 Java
Java内存管理深度解析####
本文深入探讨了Java虚拟机(JVM)中的内存分配与垃圾回收机制,揭示了其高效管理内存的奥秘。文章首先概述了JVM内存模型,随后详细阐述了堆、栈、方法区等关键区域的作用及管理策略。在垃圾回收部分,重点介绍了标记-清除、复制算法、标记-整理等多种回收算法的工作原理及其适用场景,并通过实际案例分析了不同GC策略对应用性能的影响。对于开发者而言,理解这些原理有助于编写出更加高效、稳定的Java应用程序。 ####
|
2月前
|
存储
阿里云轻量应用服务器收费标准价格表:200Mbps带宽、CPU内存及存储配置详解
阿里云香港轻量应用服务器,200Mbps带宽,免备案,支持多IP及国际线路,月租25元起,年付享8.5折优惠,适用于网站、应用等多种场景。
762 0

推荐镜像

更多
  • DNS