【C语言】memmove()函数(拷贝重叠内存块函数详解)

简介: 【C语言】memmove()函数(拷贝重叠内存块函数详解)

一.memmove()函数简介

我们先来看一下cplusplus.com - The C++ Resources Network网站上memmove()函数基本信息

1.函数功能

可以看到,memmove()函数的功能是:

源头指向的内存块拷贝固定字节数的数据目标指向的内存块,并且源头的内存块与目标内存块可以重叠.(最后一点是memmove()与memcpy最大的区别)


2.函数参数

该函数一共有三个参数,分别是:

void * memmove ( void * destination, const void * source, size_t num );

1>.void * destination

第一个参数的类型是无类型指针(void*),它指向拷贝的目的地内存块,它的作用是为函数提供目的地的内存块起始地址,以便函数能够准确地将内容拷贝到我们需要的内存空间.

2>.onst void * source

第二个参数的类型是被const修饰(const修饰的指针,const在*左边表示指针指向的内容不可修改,const在*右边表示指针的指向不可修改)无类型指针(void*),它指向拷贝数据的来源内存块,它的作用是为函数提供拷贝源头内存块起始地址,以便函数能够准确找到拷贝的源头进行拷贝.

3>.size_t num

第三个参数的类型是size_t(无符号整形),它表示要拷贝数据的字节数,它的作用是告诉函数需要拷贝的字节数是多少,以便函数精准的拷贝该数目字节数空间的内容到目的地.


3.函数返回值

函数的返回值类型是无类型指针(void*),它的作用是在函数运行结束后返回拷贝后的目的地内存块的起始地址.


4.函数头文件

该函数包含在头文件<string.h>中.


二.memmove()函数的具体使用

memmove()函数的使用场景是:

当我们想拷贝一个整型数组/结构体/枚举常量等strcpy()函数无法拷贝的数据,并且目的地内存块和源头内存块可能会有重叠的时候,我们可以考虑使用memmove()函数来完实现这一诉求,当然,想要使用memmove()函数拷贝字符串数据或者拷贝目的地内存块和源头内存块不重叠也是可以的.(但是会有些杀鸡用牛刀的感觉哈哈哈)

下面是拷贝时源内存块与目标内存块重叠的情况示意图:

1.使用memmove()函数完成拷贝整型数组数据(目的地与源重叠)

因为拷贝目的地内存块与源内存块不重叠的情况我们已经在memcpy()函数部分详细展示过了,因此在memmove()函数部分我们将着重展示它的内存块重叠时的使用情况.

如下,我们使用memmove()函数将arr数组中的1,2,3,4,5拷贝到3,4,5,6,7的位置上去:

分别给memmove()函数传入三个参数:

拷贝目的地地址(即arr+2),拷贝来源地址(即arr),拷贝字节数(即sizeof(arr[0])*5).

/* memmove 使用测试 */
#include <stdio.h>
#include <string.h>
 
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  memmove(arr+2, arr, sizeof(arr[0])*5);    
  //sizeof(arr[0])计算的结果是arr数组中一个元素的字节大小,乘5代表5个
  
  for (int i = 0; i < 10; i++)
  {
    printf("%d ", arr[i]);
  }
  return 0;
}

在vs编译器中运行查看结果:

可以看到memmove()函数成功的将arr数组中的1,2,3,4,5拷贝到了3,4,5,6,7的位置上.


2.使用memmove()函数完成拷贝字符数组数据(目的地与源重叠)

如下,我们使用memmove()函数将str数组的" very useful"拷贝到" useful......"的位置上去:

分别给memmove()函数传入三个参数:

拷贝目的地地址(即str+20),拷贝来源地址(即str+15),拷贝字节数(即11).

/* memmove 使用测试 */
#include <stdio.h>
#include <string.h>
 
int main()
{
  char str[] = "memmove can be very useful......";
  memmove(str + 20, str + 15, 11);
  //使用memmove()函数拷贝内存块重叠字符串
 
  printf("%s\n", str);
  return 0;
}

在vs编译器中运行查看结果:

可以看到memmove()函数成功的将str数组的" very useful"拷贝到了" useful......"的位置上.


三.模拟实现memmove()函数功能

🎏实现思路

1.函数参数及返回值设定逻辑

📌函数参数:
void * destination

因为memmove()函数要实现的是内存空间的拷贝,所以在使用memmove()函数时我们难免会遇到拷贝不同类型数据的可能,因此在这里我们需要将目的地的地址类型设置为无类型指针(void*),以便函数后续可以处理任意类型的数据.

const void * source

来源地址的类型设置为无类型指针(void*)的原因与目的地的原因相同,都是便于函数可以处理任意类型的数据.

而给来源的地址指针加上const的原因是防止拷贝的过程中将来源的内容不慎修改,在*指针左侧加上const就可以使const修饰的指针指向的内容变成常量.

size_t num

因为要拷贝的字节数恒为非负数,因此字节数的类型是无符号整形(size_t).

📌函数返回值:
void*

函数返回值设置为void*的原因同目的地及来源地相同,都是便于函数可以在处理完任意类型的数据后可以返回目的地的地址.


2.函数功能实现逻辑

在讲实现逻辑之前,我们先分情况讨论一下在拷贝数据时,我们可能遇到的几种情况:

注:以下演示中,*src指针指向源头起始内存块,*dest指针指向目标起始内存块.

1.情况一(两个内存块不重叠)

这种情况下内存空间的拷贝逻辑是较为简单的,不论是数据从前向后拷贝还是从后向前拷贝,结果都是正确的.

例如这种情况:

或是这种情况:


2.情况二(两个内存块有重叠)

这种情况就比较复杂了,同样分两种情况,如图:

源头指针在前时,只能从源头内存块的后端向前端移动拷贝:

拷贝逻辑示意图(序号代表拷贝次序):

目的地指针在前时,只能从源头内存块的前端向后端移动拷贝:

拷贝逻辑示意图(序号代表拷贝次序):

分析清楚上面四种不同的情况,我们就可以在代码的编写中对不同情况进行分情况讨论了.

当然这里的分情况是十分灵活的,你可以四种情况各分一种,也可以按拷贝模式来分,从前向后拷的算一种,从后向前拷的算另一种.

除了两个必须按固定方式拷贝的情况之外,剩下两种情况无论按哪种方式拷贝都行.

那么我们在这里就选择一种最简单也最容易理解的方式来模拟实现memmove()函数吧.

如果源头指针在前(小),从后向前拷贝;目的地指针在前(小),从前向后拷贝

数据在内存中的存储示意图:


🎏代码编写

清楚了不同情况的拷贝逻辑后,代码的编写就只需要按上面说的分两种情况就好了,以及,函数在实现时的其他需要注意的点我都标注在注释中了.

综上,my_memmove()函数的完整实现代码如下:

//模拟实现my_memmove()函数
 
void* my_memmove(void* destination, const void* source, size_t num)
{
  assert(destination);        //防止源头或目的地指针为NULL
  assert(source);
  void* ret = destination;
  if (source < destination)   //内存中数据的存储是由低地址到高地址的
  {
    //从后向前拷贝
    while (num--)   //以num为20为例,在num进入while循环之后,就立刻--,变成19了
    {
      *((char*)destination + num ) = *((char*)source + num );
      //这时source+num刚好指向的是源头内存块的最后一个字节
    }
  }
  else
  {
    while (num--)
    {
      //从前向后拷贝
      *((char*)destination) = *((char*)source);
      ++((char*)destination);        
      ++((char*)source);
      //这里使用后置++的话一定要给(char*)destination整体带上括号
      //否则后置++的优先级比(char*)的强制类型转换的优先级高,
      //导致指针类型还是void*时就进行++操作,这是在C标准中是不允许的
    }
  }
  return ret;
}

🎏运行测试

1.测试my_memmove()函数的逆序拷贝

如下,我们使用my_memmove()函数将arr数组的1,2,3,4,5拷贝到3,4,5,6,7的位置上:

即按照前面提到的这种情况进行拷贝:

vs2022测试运行结果:

可以看到my_memmove()函数成功的将arr数组中的1,2,3,4,5拷贝到了3,4,5,6,7的位置上.


2.测试my_memmove()函数的顺序拷贝

如下,我们使用my_memmove()函数将arr数组的3,4,5,6,7拷贝到1,2,3,4,5的位置上:

即按照前面提到的这种情况拷贝:

vs2022中测试运行结果:

可以看到my_memmove()函数成功的将arr数组中的3,4,5,6,7拷贝到了1,2,3,4,5的位置上.


结语

希望这篇memmove()函数的介绍到能对大家有所帮助,欢迎大佬们留言或私信与我交流.

最后的最后,感谢这位大佬指出了我在memcpy()函数阶段模拟实现的不足,因为和他的交流,才促成了这篇博客的产生:

学海漫浩浩,我亦苦作舟!关注我,大家一起学习,一起进步!



C语言内存相关库函数思维导图:


相关文章
|
1月前
|
存储 编译器 程序员
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
在C语言中,内存布局是程序运行时非常重要的概念。内存布局直接影响程序的性能、稳定性和安全性。理解C程序的内存布局,有助于编写更高效和可靠的代码。本文将详细介绍C程序的内存布局,包括代码段、数据段、堆、栈等部分,并提供相关的示例和应用。
48 5
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
|
1月前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
66 10
|
1月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
52 9
|
1月前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
41 8
|
1月前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
50 6
|
1月前
|
存储 C语言
【C语言】输入/输出函数详解
在C语言中,输入/输出操作是通过标准库函数来实现的。这些函数分为两类:标准输入输出函数和文件输入输出函数。
267 6
|
1月前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
64 6
|
1月前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
42 5
|
2月前
|
传感器 人工智能 物联网
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发,以及面临的挑战和未来趋势,旨在帮助读者深入了解并掌握这些关键技术。
55 6
|
2月前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
65 1