今天听到有个同学,去面试。考到了关于结构体之间内容的复制,题目大概是知道两个结构体名A,B,但不知道A,B结构体中的内容。要求你把A结构体的成员,复制到B结构体中。
在这里需要用到memcpy函数。所以我今天把相关mem函数进行一个小结。供大家参考。有什么不对的地方,请指出。
strcpy
原型:extern char *strcpy(char *dest,char *src);
功能:把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的区别的理解。我有什么遗漏吗?请指正。 不论你的水平如何,请留下你的想法。