您的位置:首页 > 其它

memset、memcpy、memmove的常见用法和区别,以及memcpy与strcpy的区别

2013-03-28 16:28 579 查看
今天听到有个同学,去面试。考到了关于结构体之间内容的复制,题目大概是知道两个结构体名A,B,但不知道A,B结构体中的内容。要求你把A结构体的成员,复制到B结构体中。

在这里需要用到memcpy函数。所以我今天把相关mem函数进行一个小结。供大家参考。有什么不对的地方,请指出。

strcpy

原型:extern char *strcpy(char *dest,char *src);

用法:#include <string.h>

功能:把src所指由NULL结束的字符串复制到dest所指的数组中。

说明:src和dest所指内存区域不可以重叠且dest必须有足够的空间来容纳src的字符串。

返回值:指向dest的指针。

例:char a[100],b[50];strcpy(a,b);

如用strcpy(b,a),要注意a中的字符串长度(第一个‘\0’之前)是否超过50位,如超过,则会造成b的内存地址溢出。

memcpy

原型:extern void *memcpy(void *dest, void *src, unsigned int count);

用法:#include <string.h>

功能:由src所指内存区域复制count个字节到dest所指内存区域。

说明:src和dest所指内存区域不能重叠,

函数返回值:指向dest的指针。可以拿它拷贝任何数据类型的对象。

举例:char a[100],b[50]; memcpy(b, a, sizeof(b));

注意如用sizeof(a),会造成b的内存地址溢出。

memset

原型:extern void *memset(void *buffer, int c, int count);

用法:#include <string.h>

功能:把buffer所指内存区域的前count个字节设置成字符c。

说明:返回指向buffer的指针。用来对一段内存空间全部设置为某个字符。

举例:char a[100];memset(a, '\0', sizeof(a));

memset可以方便的清空一个结构类型的变量或数组。

用memset就非常方便:

memset(&stTest,0,sizeof(struct sample_struct));

如果是数组:

struct sample_struct TEST[10];

则memset(TEST,0,sizeof(struct sample_struct)*10);

memmove

原型:void *memmove( void* dest, const void* src, size_tcount );

头文件:<string.h>

功能:由src所指内存区域复制count个字节到dest所指内存区域。

说明:memmove用于从src拷贝count个字符到dest,如果目标区域和源区域有重叠的话,memmove能够保证源串在被覆盖之前将重叠区域的字节拷贝到目标区域中。但复制后dest内容会被更改。但是当目标区域与源区域没有重叠则和memcpy函数功能相同。函数返回指向dest的指针

程序示例

// memmove.c

#include <stdio.h>

#include <string.h>

int main(void)

{

char s[]="Golden Global View";

memmove(s,s+7,strlen(s)+1-7);

printf("%s",s);

getchar();

return 0;

}

Memcpy与strcpy的实现和区别:

mem是对内存,str是一段字符串进行操作,不过它的长度是以‘\0’来做最后一位,随时都可以计算出来所以memcpy需要第三个参数,而strcpy不需要。

void* memcpy(void* pvTo, const void* pvForm, size_t size)

{

assert((pvTo!= NULL) && (pvFrom!= NULL));//使用断言防止传递空地址

unsigned char* pbTo = (unsigned char*)pvTo;//防止改变pvTo的地址

unsigned char* pbFrom = (unsigned char*)pvFrom;//防止改变pvFrom的地址

while(size-- > 0)

{

*pbTo++ = *pbFrom++;

}

return pvTo;

}

char* strcpy(char* pDest, const char* pSrc)

{

assert((pDest != NULL) && (pSrc != NULL));

char *pTmp = pDest;

while ((*pDest++ = *pSrc++) != '\0')

;

return pTmp;

}

memcpy和memmove 的用法和区别:

void *memmove(void *dest, const void *source, size_t count)

{

// assert((NULL != dest) && (NULL != source)); //如果它的条件返回错误,则终止程序执行

详解见文章结尾小知识点补充。

char *tmp_source = (char *)source;

char *tmp_dest = (char *)dest;

if(tmp_dest <= tmp_source || (tmp_source + count) <= tmp_dest)

{ // 如果没有重叠区域

while(count--)

*tmp_dest++ = *tmp_source++;

}

else

{ //如果有重叠

tmp_source += count - 1;

tmp_dest += count - 1;

while(count--)

*tmp_dest-- = *tmp_source--;

}

return dest;

}

函数memcpy()从source指向的区域向dest指向的区域复制count个字符,如果两数组重叠,不定义该函数的行为.而memmove(),如果两函数重叠,赋值仍正确进行。

memcpy函数假设要复制的内存区域不存在重叠,如果你能确保你进行复制操作的的内存区域没有任何重叠,可以直接用memcpy;如果你不能保证是否有重叠,为了确保复制的正确性,你必须用memmove。

【补充知识点】

assert() 宏用法

注意:assert是宏,而不是函数。在C的assert.h 头文件中。

assert宏的原型定义在<assert.h>中,其作用是如果它的条件返回错误,则终止程序执行,原型定义:

#include <assert.h>

void assert( int expression );

assert的作用是先计算表达式 expression ,如果其值为假(即为0),那么它先向标准错误流stderr打印一条出错信息,然后通过调用 abort 来终止程序运行;否则,assert()无任何作用。宏assert()一般用于确认程序的正常操作,其中表达式构造无错时才为真值。完成调试后,不必从源代码中删除assert()语句,因为宏NDEBUG有定义时,宏assert()的定义为空。

这篇文章里的内容就是我对memset、memcpy、memmove的常见用法和区别,以及memcpy与strcpy的区别的理解。我有什么遗漏吗?请指正。 不论你的水平如何,请留下你的想法。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: