C/C++中的mem函数和strcopy函数的区别和应用

2020-01-06 15:42:16于丽

strcpy和memcpy都是标准C库函数,它们有下面的特点。

strcpy提供了字符串的复制。即strcpy只用于字符串复制,并且它不仅复制字符串内容之外,还会复制字符串的结束符。

memcpy提供了一般内存的复制。即memcpy对于需要复制的内容没有限制,因此用途更广。

mem系列函数是面试的时候常考的知识点,我们需要熟练掌握这三个函数的原理和代码实现,要能准确无误的写出代码。

memcpy、memset和memset三个函数在使用过程中,均需包含以下头文件:


//在C中
#include<string.h>
//在C++中
#include<cstring>

memcpy

memcpy函数是C/C++中的内存拷贝函数,它的功能是从源src所指的内存地址的起始位置开始,拷贝n个字节到目标dst所指的内存地址的起始位置中。

研究函数功能最好的办法就是研究其源代码,这里在网上找了一份,如下:


void*__cdeclmemcpy(void* dst,constvoid* src,size_tcount)
{
void* ret = dst;
while(count--)
{ 
// 注意, memcpy函数没有处理dst和src区域是否重叠的问题
*(char*)dst = *(char*)src;
dst = (char*)dst +1;
src = (char*)src +1;
}
return(ret);
}

源代码比较简单,定义一个计数,然后从头到尾一次将src指向的值拷贝给dst,库函数中的memcpy不能处理dst和src中存在重叠部分这种情况。

那么处理重叠部分的话,我们可以采用从后往前依次拷贝的方法,下面给出我修改过的函数代码:


void*__cdeclmemcpy(void* dst,constvoid* src,size_tcount)
{
char*pDst =static_cast<char*> dst;
constchar*pSrc =static_cast<constchar*> src;
//检查参数
if(pDst==NULL|| pSrc==NULL|| count <=0){
returnNULL;
}
//判断有是否存在重叠部分
if(pDst > pSrc && pDst < pSrc + count){
for(size_ti=count-1; i>=0; i--)
{
pDest[i] = pSrc[i];
}
}
else{
for(size_ti=0; i<count; i++)
{
pDest[i] = pSrc[i];
}
}
returnpDst;
}

memset

memset一般用于对内存初始化,在这里需要注意的是,memset函数是对内存的每个字节(按字节)设置成c的值。其函数原型如下:


voidmemset(void*s,intc,size_tn)
{
constunsignedcharuc = c;//将int转换成char,截去c的高24位,留下低8位
unsignedchar*su;
for(su = s;0< n; ++su, --n)
*su = uc;
returns;
}

注意,这里有一个坑,memset一般用于将内存清零,你要是想将这段内存初始化为1而写下下面的代码:


intnum[10];
memset(num,1,sizeof(int)*10);

这里并不会如你所愿,num的每一个数都被初始化为16843009,原因就是上述提到的会截去c的高24位。

使用memset初始化比用for循环初始化要快很多,所以在初始化基本类型数据,结构体等的时候尽量选择memset,memset可以方便的清空一个结构类型的变量或数组。