memcpy的使用方法总结

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的实现过程,里面是一个一个字符的赋值的,想连续赋值,就要把指针指向连续的内存的首地址,所以,
真的非常不好表达,呵呵,就这样了,一大推零散的知识。。。

时间: 2024-12-08 09:46:33

memcpy的使用方法总结的相关文章

Cortex-A8处理器memcpy的优化方案

公司产品涉及到视频编解码和人脸识别,但是出现帧率太低的现象,同事做了一些测试,最后问题定位到应用程序中memcpy慢,特别是由uncached区域(视频采集buf,使用mmap对/dev/mem映射到用户空间)到cached区域(用户空间malloc),因此需要想办法进行下优化. 首先交代下设备处理器背景,处理器是公司自研,使用ARM Cortex-A8处理器核,CPU为800MHZ,SAXI总线为533MHZ.后续一系列测试都是在相同的clk下进行,保证测试数据的硬件环境一致,有可对比性. 同

C++:怎样把一个int转成4个字节?

大家都知道,一个int 或 unsigned int是由4个字节组成的,(<C/C++学习指南>,第3章,第3.2.3节:变量的内存视图) 比如, int   n  =  sizeof( int) ;   // n为4 大家也可以清晰明确地看到内存里,那4个字节就在那,(<C/C++学习指南>,附录:VC2008调试方法) 但问题是:怎样在代码里转成4个字节呢? 方法1:memcpy 这样的方法非常暴力,也不科学.大家先试一下. unsigned int a = 0x1234567

memcpy函数的使用方法

c和c++使用的内存拷贝函数,memcpy函数的功能是从源src所指的内存地址的起始位置开始拷贝n个字节到目标dest所指的内存地址的起始位置中. 1.函数原型 void *memcpy(void *dest, const void *src, size_t n); 2.功能 从源src所指的内存地址的起始位置开始拷贝n个字节到目标dest所指的内存地址的起始位置中 3.所需头文件 C语言中使用#include <string.h>; C++中使用#include <cstring>

strcpy、strncpy与memcpy的区别与使用方法

今天做题用到了这两个函数,不是很懂,学习了下~ 转自http://www.cnblogs.com/houjun/p/4913216.html 本文参考<C 标准库>编写. 一.函数说明 1.memcpy函数 void  *memcpy(void *s1,  const void *s2,  size_t  n); 说明: 函数memcpy从s2指向的对象中复制n个字符到s1指向的对象中.如果复制发生在两个重叠的对象中,则这种行为未定义. 返回值: 函数memcpy返回s1的值. 2.strcp

strcpy和memcpy的区别

strcpy提供了字符串的复制.即strcpy只用于字符串复制,并且它不仅复制字符串内容之外,还会复制字符串的结束符. 已知strcpy函数的原型是:char* strcpy(char* dest, const char* src);memcpy提供了一般内存的复制.即memcpy对于需要复制的内容没有限制,因此用途更广. char * strcpy(char * dest, const char * src) // 实现src到dest的复制 { if ((src == NULL) || (d

memcpy与memmove

函数原型: void* memcpy(void *dst,void const *src,size_t count) void* memmove(void *dst,void const *src,size_t count) 头文件: #include<string.h> 引入: 字符串由'\0'结尾,所以字符串内部不能包含任何'\0'字符('\0'的ASCII值为0),否则我们将读不到'\0'后的字符内容.但是,非字符串内部包含零值的情况并不罕见,我们无法利于其它字符串函数来处理这类数据.所

strcpy, memcpy, memset函数

一. strcpy函数 原型声明:char *strcpy(char* dest, const char *src); 头文件:#include <string.h> 和 #include <stdio.h> 功能:把从src地址开始且含有NULL结束符的字符串复制到以dest开始的地址空间 说明:src和dest所指内存区域不可以重叠且dest必须有足够的空间来容纳src的字符串. 返回指向dest的指针. 实现代码: char * strcpy(char * strDest,c

JNI学习2:android 调用C语言方法与C语言调用android方法

#include <jni.h> #include <stdio.h> #include <stdlib.h> #include <jni.h> #include <android/log.h> #define LOG_TAG "System.out" //日志乱码时请将项目文件编码设置为UTF-8 /**debug级别日志:ANDROID_LOG_DEBUG:级别,LOG_TAG:标签,__VA_ARGS__:日志内容)**

strcpy和memcpy的编程实现

1 char *strcpy(char *dest,char *src) 2 { 3 char *d=dest;//输入的参数进行备份 4 char *s=src; 5 6 int count=0; 7 8 assert(dest!=NULL && src!=NULL);//指针合法性检测 if(src==dest) return src; 11 count = strlen(src) +1;//计算src字符串长度 12 if(count <= 1) return 0;//src字