memcpy的用法总结

转载:https://blog.csdn.net/qq_35040828/article/details/71123521
memcpy的用法总结
原型:extern void *memcpy(void *dest, void *src, unsigned int count);

用法:#include <string.h>

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

说明:src和dest所指内存区域不能重叠,函数返回指向dest的指针。

举例:

  // memcpy.c
  
  #include <syslib.h>
  #include <string.h>

  main()
  {
    char *s="Golden Global View";
    char d[20];
    
    clrscr();
    
    memcpy(d,s,strlen(s));
    d[strlen(s)]=0;
    printf("%s",d);

    getchar();
    return 0;
  }

1、memcpy 函数用于 把资源内存(src所指向的内存区域) 拷贝到目标内存(dest所指向的内存区域);拷贝多少个?有一个size变量控制
拷贝的字节数;
函数原型:void memcpy(void dest, void src, unsigned int count);
用法:(1)可以拷贝任何类型的对象,因为函数的参数类型是void
(未定义类型指针),也就是说传进去的实参可以是int
,short
,char等等,
但是由于函数拷贝的过程是一个字节一个字节的拷贝的,所以实际操作的时候要把void
强制转化为char*,这样在指针加的时候才会保证每次加一个字节,呵呵
函数源代码实现:
void *memcpy1(void *desc,const void * src,size_t size)
{
if((desc == NULL) && (src == NULL))
{
return NULL;
}
unsigned char desc1 = (unsigned char)desc;
unsigned char src1 = (unsigned char)src;
while(size-- >0)
{
*desc1 = *src1;
desc1++;
src1++;
}
return desc;
}

int _tmain(int argc, _TCHAR* argv[])
{
int dest[2] = {0};
const char src[5] = “1234”;
//printf(src);
memcpy1(dest,src,sizeof(src));
//*(dest+5) = ‘/0’;
printf((char )dest);
int m = -1;
return 0;
}
注意事项:(1)void
一定要返回一个值(指针),这个和void不太一样!
(2)首先要判断指针的值不能为空,desc为空的话肯定不能拷贝内存空间,src为空相当于没有拷贝;所以之间return掉;
(3)""空串是指内容为0,NULL是0,不是串;两个不等价;
(4)int dest[2] = {0};这是对int 类型的数组初始化的方法;如果是char类型,就用char a[5] = “1234”; 注意数组下标要
多于实际看到的字符数,因为还有’/0’
(5)printf((char )dest);这句话,是把 char 类型 src 传到 int 类型的 dest的内存强制转化成char类型,然后打印出来;
因为直接看int类型的dest是看不到里面的内容的;因为有unsigned char desc1 = (unsigned char)desc;所以字符可以传
到dest里面保存起来,dest所指向的内存长度4个字节,强制转化为char 就是把四个字节分成一个一个的字节,这样就可以看到
一个个字符了,如果定义成char dest[5] = “1234”;就不用转化;呵呵,表达起来真累人;
(6)memcpy1(dest,src,sizeof(src));注意里面的sizeof(src),这个是包括字符串的结束符’/0’的;所以不用担心printf(dest);
但是如果用memcpy1(dest,src,4);没有’/0’就要
(dest+5) = ‘/0’;这样保证是一个完整的字符串;
(7)如果初始化的时候:
char dest[1024] = “12345666”;//{0};
const char src[5] = “3333”;
那么拷贝的时候,如果用memcpy1(dest,src,sizeof(src));则printf(dest);出来是3333
如果memcpy1(dest,src,4);则printf(dest);出来是33335666;因为上面的sizeof(src),包含’/0’,所以拷贝过去的字符串以’/0’
结束,就只有3333,而如果传4个字符,’/0’是第五个字符,那就遇到dest[1024] 的’/0’结束,所以是33335666
字符串的’/0’问题一定要注意啊!!!

实际应用:
unsigned char g_pData[1024] = “”;
DWORD g_dwOffset = 0;
bool PackDatatoServer(const unsigned char *pData, const unsigned int uSize)
{
memcpy(g_pData+g_dwOffset, pData, uSize);
g_dwOffset += uSize;
//g_pData += uSize;
return true;
}

void main()
{
const unsigned char a[4] = “123”;
PackDatatoServer(a, 3);
PackDatatoServer(a, 1111);
int b = -1;
}

PackDatatoServer()函数的作用是把每次的资源内存拷贝到目标内存里面,而且是累加的拷贝;也就是后一次紧接着上一次的拷贝;
显然用到了memcpy函数;
实现原理是用到了一个全局变量g_dwOffset 保存之前拷贝的长度,最开始没有想到这一点,结果每次拷贝都是一次性的,下一次拷贝把
上一次的冲掉了;所以用全局变量记录拷贝的长度;
第二个需要注意的是,拷贝的过程中注意不要改变目标指针的指向,即目标指针始终指向初始化的时候指向的位置;那么怎么实现累积拷贝呢?
就是用的指针偏移;第一次实现的时候,把g_pData += uSize;写到了函数里面,这样写是能够实现指针位移的目标,但是指针指向也发生改变;
另外:g_pData += uSize;也有报错:left operand must be l-value,原因是:把地址赋值给一个不可更改的指针!
比如:
char a[100];
char *p = new char[10];
a = p; //这里出错,注意了:数组的首地址也是一个常量指针,指向固定不能乱改的~~

char * const pp = new char[1];
pp = a; //也错
所以既不能改变首地址,又要满足累积赋值(就是赋值的时候要从赋过值的地方开始向下一个内存块赋值,想到指针加),所以想到把指针加写到
函数参数里面,这时就要充分了解memcpy的实现过程,里面是一个一个字符的赋值的,想连续赋值,就要把指针指向连续的内存的首地址,所以,

真的很不好表达,呵呵,就这样了,一大推零散的知识。。。

函数原型

void memcpy(voiddest, const void *src, size_t n);

功能

由src指向地址为起始地址的连续n个字节的数据复制到以destin指向地址为起始地址的空间内。

头文件

#include<string.h>

返回值

函数返回一个指向dest的指针。

说明

1.source和destin所指内存区域不能重叠,函数返回指向destin的指针。

2.与strcpy相比,memcpy并不是遇到’\0’就结束,而是一定会拷贝完n个字节。

memcpy用来做内存拷贝,你可以拿它拷贝任何数据类型的对象,可以指定拷贝的数据长度;

例:

char a[100], b[50];

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

strcpy就只能拷贝字符串了,它遇到’\0’就结束拷贝;例:

char a[100], b[50];

strcpy(a,b);

3.如果目标数组destin本身已有数据,执行memcpy()后,将覆盖原有数据(最多覆盖n)。如果要追加数据,则每次执行memcpy后,要将目标数组地址增加到你要追加数据的地址。

//注意,source和destin都不一定是数组,任意的可读写的空间均可。

程序例

example1

作用:将s中的字符串复制到字符数组d中。

//memcpy.c

#include<stdio.h>

#include<string.h>

intmain()

{

char*s=“Golden Global View”;

chard[20];

clrscr();

memcpy(d,s,strlen(s));

d[strlen(s)]=’\0’;//因为从d[0]开始复制,总长度为strlen(s),d[strlen(s)]置为结束符

printf("%s",d);

getchar();

return0;

}

输出结果:GoldenGlobal View

example2

作用:将s中第14个字符开始的4个连续字符复制到d中。(从0开始)

#include<string.h>

intmain()

{

char*s=“Golden Global View”;

chard[20];

memcpy(d,s+14,4);//从第14个字符(V)开始复制,连续复制4个字符(View)

//memcpy(d,s+14sizeof(char),4sizeof(char));也可

d[4]=’\0’;

printf("%s",d);

getchar();

return0;

}

输出结果: View

example3

作用:复制后覆盖原有部分数据

#include<stdio.h>

#include<string.h>

intmain(void)

{

charsrc[] = “******************************”;

chardest[] = “abcdefghijlkmnopqrstuvwxyz0123as6”;

printf(“destinationbefore memcpy: %s\n”, dest);

memcpy(dest,src, strlen(src));

printf(“destinationafter memcpy: %s\n”, dest);

return0;

}

输出结果:

destinationbefore memcpy:abcdefghijlkmnopqrstuvwxyz0123as6

destinationafter memcpy: ******************************as6

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值