C/C++之Memcpy and memmove

memcpy与memmove的目的都是将N个字节的源内存地址的内容拷贝到目标内存地址中。

但当源内存和目标内存存在重叠时,memcpy会出现错误,而memmove能正确地实施拷贝,但这也增加了一点点开销。

memmove的处理措施:

(1)当源内存的首地址等于目标内存的首地址时,不进行任何拷贝

(2)当源内存的首地址大于目标内存的首地址时,实行正向拷贝

(3)当源内存的首地址小于目标内存的首地址时,实行反向拷贝

-- memcpy实现

1 voidmemcpy(void* dest, const void* src, size_t n)
2 {
3 char* d = (char*) dest;
4 const char* s = (const char*) src;
5 while(n-–)
6 *d++ = *s++;
7 return dest;
8 }

 

Notes:

  • memcpy的参数指针类型是void*,具体赋值操作是以字节为单位。
  • 必须进行类型转换。
  • 返回的还是void*型的dest。

-- memmove实现

01 voidmemmove(void* dest, const void* src, size_t n)
02 {
03 char* d = (char*) dest;
04 const char* s = (const char*) src;
05  
06 if (s>d)
07 {
08 // start at beginning of s
09 while (n--)
10 *d++ = *s++;
11 }
12 else if (s<d)
13 {
14 // start at end of s
15 d = d+n-1;
16 s = s+n-1;
17  
18 while (n--)
19 *d-- = *s--;
20 }
21 return dest;
22 }

 示意图:

(1)内存低端 <-----s-----> <-----d-----> 内存高端 start at end of s(2)内存低端 <-----s--<==>--d----->      内存高端 start at end of s(3)内存低端 <-----sd----->              内存高端 do nothing(4)内存低端 <-----d--<==>--s----->      内存高端 start at beginning of s(5)内存低端 <-----d-----> <-----s-----> 内存高端 start at beginning of s

Notes:
  • s==d时,什么都不做。
  • d在前,正向拷贝。
  • d在后,逆向拷贝。

1.memmove

函数原型:void *memmove(void *dest, const void *source, size_t count)

返回值说明:返回指向dest的void *指针

参数说明:dest,source分别为目标串和源串的首地址。count为要移动的字符的个数

函数说明:memmove用于从source拷贝count个字符到dest,如果目标区域和源区域有重叠的话,memmove能够保证源串在被覆盖之前将重叠区域的字节拷贝到目标区域中。

2.memcpy

函数原型:void *memcpy(void *dest, const void *source, size_t count);

返回值说明:返回指向dest的void *指针

函数说明:memcpy功能和memmove相同,但是memcpy中dest和source中的区域不能重叠,否则会出现未知结果。
原型:extern char *strcpy(char *dest,char *src);  功能:把src所指由NULL结束的字符串复制到dest所指的数组中。  说明:src和dest所指内存区域不可以重叠且dest必须有足够的空间来容纳src的字符串,返回指向dest的指针。

  其实在strcpy的实现比较多,但思想是一致的,一般用C来实现,但是memcpy和memmove这样的函数可能是用汇编实现的,并且充分利用块拷贝的思想,不会单字节单字节的拷贝。所以效率strcpy<memcpy.

memmove一般由于要判断内存是否重合,效率也会较memcpy低些。

1 /***
2 * @brief 以字节的方式直接拷贝
3 * 库中实现是以汇编实现,
4 * 其实可以直接调用strncat函数
5 * **/
6 void *memcpy(void *dst,void *src,size_t n)
7 {
8 char *dp = (char *)dst;
9 char *sp = (char *)src;
10 assert((src!=0)&&(dst!=0)&&(n>0));//not null
11 while(n--)
12 *(dp++) = *(sp++);
13 /**!边界*/
14 dp = ‘\0‘;
15 return dst;
16 }

memmove

 1 void *memmove(void *dst,const void *src,int n)                                                 2 {                                                                                              3     char *dp = (char *)dst;                                                                    4     char *sp = (char *)src;                                                                    5     assert((src!=0)&&(dst!=0)&&(n>0));//not null                                              6     //非重叠                                                                                   7     //dp < sp                                                                                  8     //dp > (sp+n)                                                                              9     if(dp<sp||(sp+n)>=dp)                                                                     10     {                                                                                         11         while(n--)                                                                            12             *(dp++) = *(sp++);                                                                13         *dp = ‘\0‘;                                                                           14     }else if(sp<dp)//重叠 (此时条件 sp<dp<(sp+n))如果sp==dp则快速的返回                       15     {//反向拷贝                                                                               16         sp += n;                                                                              17         dp += n;                                                                              18         *dp = ‘\0‘;                                                                           19         while(n--)                                                                            20             *(--dp) = *(--sp);                                                                21     }                                                                                         22     return dst;                                                                               23 }            

 注意对于重合的要反向拷贝



/*
 *Magicman
 *myMemcpy.c
 *不调用库函数,实现内存拷贝
*/

#include <stdio.h>
#include <assert.h>
#include <stdlib.h>

void *myMemcpy(void *dest, const void *src, int len)
{
    assert((dest != NULL) && (src != NULL) && (len >= 0));

    if (dest == src)
    {
        return dest;
    }

    while (len-- > 0)
    {
        *(char *)dest++ = *(char *)src++;
    }

    return dest;
}

int main(int argc, char argv[])
{
    char str[20] = "Testing myMemory!";
    char pstr[20] = "";
    char *pp = str;
    int ia[10] = {1,2,3,4,5,6,7,8,9,10};
    int ib[10] = {};
    int *ip = NULL;

myMemcpy((void *)pstr, (void *)str, sizeof(str));

printf("%s/n", pstr);

printf("%s/n", myMemcpy((void *)pp, (void *)str, 20));

myMemcpy((void *)ib, (void *)ia, 5*sizeof(int));

for (ip = ib; ip < ib + 10; ip++)
    {
        printf("%d  ", *ip);
    }

printf("/n");
    
    return 0;
}



让自己实现memcpy库函数,要求考虑特殊情况,两段内存存在覆盖,以及指针为空的情况。

几点结论: 
1,memcpy实现从源source中拷贝n个字节到目标destin中,src源数据应该保留。
2,memmove实现移动一块字节,src源数据可以不保留。
3,memcpy没有考虑内存覆盖问题(由assert条件可知);而memmove考虑了内存覆盖问题,并给出了解决办法。
4,memcpy和memmove中不需要考虑数组越界问题,dst的长度应该大于src的长度,这是调用者应该考虑的问题。

时间: 2024-10-10 07:11:52

C/C++之Memcpy and memmove的相关文章

实现memcpy和memmove函数

题目: 自己定义一个函数,实现my_memcpy和my_memmove函数. 题目分析: memcpy函数主要实现的是内存的拷贝,函数接受任意类型的参数,并且有拷贝个数的限制,函数与strcpy函数在功能上有相似点,也有不同点.memmove函数在memcpy函数的基础上解决了内存重叠的问题.下面是memcpy和memmove函数的实现: my_memmove函数: #define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> #include &l

memcpy 与 memmove

memcpy:C和C++当中使用的内存拷贝函数,memcpy函数的功能是从源src所指的内存地址的起始位置开始拷贝n个字节到目标dest所指的内存地址的起始位置中. 函数原型 void* memcpy(void* dest, const void* src, size_t count) 函数返回指向dest的指针: mencpy与memmove的区别:根据源代码编写自己的Mencpy 与Memmove #include <iostream> using namespace std; #incl

memcpy、memmove、memset及strcpy函数实现和理解

memcpy.memmove.memset及strcpy函数实现和理解 关于memcpy memcpy是C和C++ 中的内存拷贝函数,在C中所需的头文件是#include<string.h>, 在C++中需要包含的头文件是#include其函数原型如下: void *memcpy(void *dest, const void *src, size_t n); 其功能是从源src所指的内存地址的起始位置开始拷贝n个字节到目标dest所指的内存地址的起始位置中. 库函数中的memcpy不能处理sr

C语言实现memcpy和memmove

0.两者比较: memmove用于从src拷贝count个字符到dest,如果目标区域和源区域有重叠的话,memmove能够保证源串在被覆盖之前将重叠区域的字节拷贝到目标区域中.但复制后src内容会被更改.但是当目标区域与源区域没有重叠则和memcpy函数功能相同. memmove在copy两个有重叠区域的内存时可以保证copy的正确,而memcopy就不行了,但memcopy比memmove的速度要快一些,如:char s[] = "1234567890";char* p1 = s;

memcpy、memmove、memset、memchr、memcmp、strstr详解

第一部分 综述 memcpy.memmove.memset.memchr.memcmp都是C语言中的库函数,在头文件string.h中.memcpy和memmove的作用是拷贝一定长度的内存的内容,memset用于缓冲区的填充工作,memchr用于字符的查找工作,memcmp用于比较内存中缓冲区的大小. 第二部分  介绍 1.memcpy和memmove memcpy()--拷贝内存内容 表头文件:#include<string.h>或#include<cstring> 定义函数:

memcpy vs memmove

[本文连接] http://www.cnblogs.com/hellogiser/p/memcpy_vs_memmove.html [分析] memcpy与memmove的目的都是将N个字节的源内存地址的内容拷贝到目标内存地址中. 但当源内存和目标内存存在重叠(memory overlapping)时,memcpy会出现错误,而memmove能正确地实施拷贝,但这也增加了一点点开销. memmove的处理措施: (1)当源内存的首地址等于目标内存的首地址时,不进行任何拷贝 (2)当源内存的首地址

memcpy和memmove的区别

函数memcpy()   从source  指向的区域向dest指向的区域复制count个字符,如果两数组重叠,不定义该函数的行为.     而memmove(),如果两函数重叠,赋值仍正确进行. memcpy函数假设要复制的内存区域不存在重叠,如果你能确保你进行复制操作的的内存区域没有任何重叠,可以直接用memcpy:     如果你不能保证是否有重叠,为了确保复制的正确性,你必须用memmove. memcpy和memmove的区别

关于memcpy和memmove的一点说明

今天看到书上降到memcpy和memmove的区别才突然 发现原来两者之间有如此区别,以前只知道这两个函数是 实现同样的功能,没有接触到其不同. memcpy和memmove在MSDN的定义如下: 从两者的声明来看的确没有区别,我们来看这样一个例子 当我们需要将char* src="abcde"这个字符串全部copy到dest中 然而src与dest在内存中大概是这样存在的: 内存地址   低------>高   1 src dest 2 1 2 3 4 5 6 3 [ a ][

[整理]内存重叠之memcpy、memmove

函数原型: void *memcpy( void *dest, const void *src, size_t count ); void *memmove( void* dest, const void* src, size_t count );  1.memcpy和memmove相同点都是用于从src拷贝count个字节到dest. 2.memcpy和memmove区别如果目标区域和源区域有重叠的话:memcpy不能够确保源串所在重叠区域在拷贝之前被覆盖.memmove能够保证源串在被覆盖之