关于字串拷贝(strcpy之类)和内存拷贝(memcpy)

4 篇文章 0 订阅
3 篇文章 0 订阅

先说一下关于memcpy,有参考http://blog.csdn.net/xiaobo620/article/details/7488827,是直接拷贝内存,因而不会有类型限制,最简单的实现就是:

void *mymemcpy(void *dst,const void *src,size_t num)
{
	assert((dst!=NULL)&&(src!=NULL));
          //assert(des>=src+num||src>dst+num);
	byte * psrc = (byte *)src;//byte 既为unsigned char类型
	byte * pdst = (byte *)dst;
	while(num-->0)*pdst++ = *psrc++;
	return dst;
}

但是这种写法,并没有考虑到两个拷贝内存的重叠的问题,所以,进一步的改进,(通过从尾部读起可解决):

void * mymemcpy(void *dest, const void *src, size_t count)
{
    if (dest == NULL || src == NULL)
          return NULL;
    char *pdest = static_cast <char*>(dest);
    const char *psrc  = static_cast <const char*>(psrc);
    int n = count;
    
    if (pdest > psrc && pdest < psrc+count)
    {
        for (size_t i=n-1; i != -1; --i)
        {
                pdest[i] = psrc[i];
        }
    }
    else
    {
        for (size_t i= 0; i < n; i++)
        {
                pdest[i] = psrc[i];
        }
    }
    
    return dest;
}
如果加之最求效率,可以通过字长来拷贝,不对齐的部分在通过字节拷贝:

void *mymemcpy(void *dst,const void *src,size_t num)
{
	assert((dst!=NULL)&&(src!=NULL));
	int wordnum = num/4;//计算有多少个32位,按4字节拷贝
	int slice = num%4;//剩余的按字节拷贝
	int * pintsrc = (int *)src;
	int * pintdst = (int *)dst;
	while(wordnum--)*pintdst++ = *pintsrc++;
	while (slice--)*((char *)pintdst++) =*((char *)pintsrc++);
	return dst;
}
默认的memcpy是 没有考虑重叠的,另外的一个函数memmov则是考虑了这个情况,具体实现就如同上面源码类似。

下来就说一下字串的复制,当然memcpy就行,他很一般比如strcpy的区别就是:(参考 http://www.cnblogs.com/stoneJin/archive/2011/09/16/2179248.htmlhttp://blog.csdn.net/zzy7075/article/details/7835361

1、复制的内容不同。strcpy只能复制字符串,而memcpy可以复制任意内容,例如字符数组、整型、结构体、类等。
2、复制的方法不同。strcpy不需要指定长度,它遇到被复制字符的串结束符"\0"才结束,所以容易溢出。memcpy则是根据其第3个参数决定复制的长度。
3、用途不同。通常在复制字符串时用strcpy,而需要复制其他类型数据时则一般用memcpy

实现也是简单的了:

char * strcpy(char * dest, const char * src) // 实现src到dest的复制
{
  if ((src == NULL) || (dest == NULL)) //判断参数src和dest的有效性
  {
 
      return NULL;
  }
  char *strdest = dest;        //保存目标字符串的首地址
  while ((*strDest++ = *strSrc++)!='\0'); //把src字符串的内容复制到dest下
  return strdest;
}
还有就是strncpy字串复制 :char * strncpy(char *dest, char *src, size_t n);
  功能:将字符串src中最多n个字符复制到字符数组dest中(它并不像strcpy一样遇到NULL才停止复制,而是等凑够n个字符才开始复制),返回指向dest的指针。
  说明:
  如果n > dest串长度,dest栈空间溢出产生崩溃异常。
  否则:
  1)src串长度<=dest串长度,(这里的串长度包含串尾NULL字符)
  如果n=(0, src串长度),src的前n个字符复制到dest中。但是由于没有NULL字符,所以直接访问dest串会发生栈溢出的异常情况。
  如果n = src串长度,与strcpy一致。
  如果n = dest串长度,[0,src串长度]处存放于desk字串,(src串长度, dest串长度]处存放NULL。
  2)src串长度>dest串长度
  如果n =dest串长度,则dest串没有NULL字符,会导致输出会有乱码。如果不考虑src串复制完整性,可以将dest最后一字符置为NULL。
  综上,( 一般正常情况下不会补\0,只有在n>src时,不足的位数会进行补\0)一般情况下,使用strncpy时,建议将n置为dest串长度(除非你将多个src串都复制到dest数组,并且从dest尾部反向操作),复制完毕后,为保险起见,将dest串最后一字符置NULL,避免发生在第2)种情况下的输出乱码问题。当然喽,无论是strcpy还是strncpy,保证src串长度<dest串长度才是最重要的。

还有就是sprintf和snprintf了:

sprintf:把格式化的数据写入某个字符串, 可以连接多个不同类型的字符串

  头文件:stdio.h函数原型:int sprintf( char *buffer, const char *format [, argument] … );
  返回值:字符串长度(strlen)

一般就是类似于 sprintf(dest,"%s",src)之类的了。

最后snprintf:

int snprintf(char *restrict buf, size_t n, const char * restrict format, ...);
函数说明:最多从源串中拷贝n-1个字符到目标串中,然后再在后面加一个0。所以如果目标串的大小为n
的话,将不会溢出。若成功则返回欲写入的字符串长度,若出错则返回负值。

就这样现了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值