memcpy、memmove、memset及strcpy函数实现和理解

关于memcpy

memcpy是C和C++ 中的内存拷贝函数,在C中所需的头文件是#include<string.h>, 在C++中需要包含的头文件是#include其函数原型如下:
void *memcpy(void *dest, const void *src, size_t n);
其功能是从源src所指的内存地址的起始位置开始拷贝n个字节到目标dest所指的内存地址的起始位置中。
库函数中的memcpy不能处理src和dest有重叠的情况。

实现思路

如果是自己实现memcpy的话,最好还是把地址有重叠的情况考虑进去,这样才能更好的体现编码的严谨。
为了更好的理解源地址和目的地址有重叠的时候,正向复制会出现的问题,下面以图形方式进行解释:

再给一个更加直接的例子,如下图:

memcpy实现代码

void *memcpy(void *dest,const void *src,size_t count)
{
char *pDest=static_cast<char *>(dest);
const char *pSrc=static_cast<const char *>(src);
//参数检查
if ( NULL == dest || NULL == src || count<=0 )
{
return NULL;
}
if( pDest>pSrc && pDest<pSrc+count )
{
for(size_t i=count-1; i>=0; i--)
{
pDest[i] = pSrc[i];
}
}
else
{
for(size_t i=0; i<count; i++)
{
pDest[i] = pSrc[i];
}
}
return pDest;
}

关于memmove

memmove和memcpy函数一样,使用时需要包含C++的#include头文件。它与memcpy的功能相似,都是将src所指的n个字节复制到dest所指的内存地址起始位置中,但该函数可以处理src和dest有重叠的情况。实际上,memcpy可以看作是memmove的子集。其代码如下(由于对memcpy做了优化,让memmove的实现和memcpy一样了。。。)

void * memmove( void * const dest, const char * const src, size_t count )
{
// 参数检查
if( NULL == dest || NULL == src || count<=0 )
{
return NULL;
} char * pSrc = (char *)src;
char * pDest = (char *)dest;
if(pDest > Psrc && pDest < Psrc + count )
{
for( size_t i = count-1; i>=0; i--)
{
pDest[i] = pSrc[i];
}
} else
{
for(size_t i=0; i<count; i++)
{
pDest[i]=pSrc[i];
}
}
return pDest;
}

关于memset

函数原型如下:
void *memset(void *s, int ch, size_t n);
该函数的功能是将s中的前n个字节用ch替换,并返回s
memset:作用是在一段内存块中填充某个给定的值,它是对较大的结构体或数组进行清零操作的一种最快方法。
在程序中给数组初始化时,常常会用到memset函数,一般的写法如下:

    int a[2];
memset(a, 0, sizeof(a));
memset(a, 0, 2*sizeof(int));

需要注意的是,赋值是逐字节进行的:
(1)若s指向char型地址,ch可为任意字符值;

(2)若s指向非char型,如int型地址,要想赋值正确,ch的值只能是-1或0,因为-1和0转化成二进制后每一位都是一样的
错误的赋值例子如下:

    int a[2];
memset(a, 1, 2*sizeof(int));

输出数组a的结果如下图:

很明显,这不是我们希望看到的,因此在用memset给数组幅初始值的时候,一般都是幅为0.
实现代码如下:

void *my_memset(void* s, int ch, size_t n)
{
//参数检查
if ( NULL == s || n<= 0)
{
return NULL;
} char *pS = (char *)s;
for ( size_t i = 0; i<n; i++)
{
pS[i] = ch;
} return pS;
}

关于strcpy

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

char *strcpy( char *dest, const char* src)
{
//参数检查
if ( NULL == dest || NULL == src )
{
return NULL;
} char *pDest = dest;
while ( *src != '\0')
{
*pDest++ = *src++;
}
*pDest = '\0'; return pDest;
}

strcpy和memcpy主要有以下3方面的区别。

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

总结

这一系列的函数都和指针的操作密切相关,如果要更好的理解C++,必须要正式指针,了解指针的特性,学会灵活的应用指针,让指针为自己的编码服务!

对于内存操作的一些列函数来说,其参数涵返回类型都是void*,而不是某种具体的类型,这样是为了任何类型的指针都可以传入memcpy和memset中,都能正确赋值。这也真实地体现了内存操作函数的意义,因为它操作的对象仅仅是一片内存,而不论这片内存是什么类型。

最后,很重要的一点是,如果对指针的位置不清楚的时候,可以借助画图来帮助理解,图形化的解释能够更好的帮助理解。

memcpy、memmove、memset及strcpy函数实现和理解的更多相关文章

  1. 走进C标准库(7)——"string.h"中函数的实现memcmp,memcpy,memmove,memset

    我的memcmp: int memcmp(void *buf1, void *buf2, unsigned int count){ int reval; while(count && ...

  2. 逆向 string.h 函数库 memset、strcpy、strcmp 函数

    memset 函数 函数原型:void *memset(void *str, int c, size_t n) 主要功能:复制字符 c(一个无符号字符)到参数 str 所指向的字符串的前 n 个字符 ...

  3. 自己实现内存操作函数memset(),memcmp(),memcpy(),memmove()

    1.memset()内存设置函数(初始化) void *my_memset(void* dest, int c, size_t count) { assert(dest != NULL); char  ...

  4. strcpy和memcpy,memmove函数的区别

    strcpy和memcpy的区别 strcpy和memcpy都是标准C库函数,它们有下面的特点. strcpy提供了字符串的复制.即strcpy只用于字符串复制,并且它不仅复制字符串内容之外,还会复制 ...

  5. strcpy函数;memcpy函数;memmove函数

    strcpy函数实现: char* strcpy(char* des,const char* source) { char* r=des; assert((des != NULL) && ...

  6. 自己实现的库函数2(memset,memcmp,memcpy,memmove)

    memset,memcmp,memcpy,memmove是对内存进行管理的库函数,为了更好的理解和使用这几个函数,自己用C语言实现一下~ //内存设置函数void* my_memset(void* d ...

  7. strcpy、memcpy和memset的区别

    strcpy 原型:extern char *strcpy(char *dest,char *src); 用法:#include <string.h> 功能:把src所指由NULL结束的字 ...

  8. strcpy,memcpy,memmove和内存重叠分析

    一:strcpy函数用法和实现: /* GNU-C中的实现(节选): */ char* strcpy(char *d, const char *s) { char *r=d; while((*d++= ...

  9. memset memcmp memcpy memmove 自己实现

    memset memcmp memcpy memmove 自己实现 memset #include <stdio.h> #include <memory.h> #include ...

随机推荐

  1. 李洪强iOS开发之OC语言类的深入和分类

    OC语言类的深入和分类 一.分类 (一)分类的基本知识  概念:Category  分类是OC特有的语言,依赖于类. 分类的作用:在不改变原来的类内容的基础上,为类增加一些方法. 添加一个分类: 文件 ...

  2. ApplePay扩大全球发卡行合作,“苹果税”撑不住了?

    5月11日Apple Pay全面登陆加拿大地区,更为重要的是,苹果终于在一些地区,开始和美国运通之外的发卡行达成了合作.这对于老是因为分账问题不愿意走出下一步的Apple Pay来说,已经是巨大的进步 ...

  3. C# Java间进行RSA加密解密交互(二)

    原文:C# Java间进行RSA加密解密交互(二) 接着前面一篇文章C# Java间进行RSA加密解密交互,继续探讨这个问题. 在前面,虽然已经实现了C# Java间进行RSA加密解密交互,但是还是与 ...

  4. Quartz任务调度快速入门

    Quartz任务调度快速入门 概述 了解Quartz体系结构 Quartz对任务调度的领域问题进行了高度的抽象,提出了调度器.任务和触发器这3个核心的概念,并在org.quartz通过接口和类对重要的 ...

  5. Django admin的一些有用定制

    Model实例,myapp/models.py: from django.db import models class Blog(models.Model): name = models.CharFi ...

  6. P25、面试题1:赋值运算符函数

    题目:如下为类型CMyString的声明,请为该类型添加赋值运算符函数. class CMyString { public: CMyString(char* pData = NULL); CMyStr ...

  7. Java基础——关键字

    volatile 用volatile修饰的变量,线程在每次使用变量的时候,都会读取变量修改后的最的值.volatile很容易被误用,用来进行原子性操作. 对于volatile修饰的变量,jvm虚拟机只 ...

  8. 用paint 计算字符串的像素宽度

    方法1: //直接返回参数字符串所占用的像素宽度 Paint paint = new Paint(); width = paint.measureText(str); 有一些view可以直接得到pai ...

  9. [POJ3352]Road Construction(缩点,割边,桥,环)

    题目链接:http://poj.org/problem?id=3352 给一个图,问加多少条边可以干掉所有的桥. 先找环,然后缩点.标记对应环的度,接着找桥.写几个例子就能知道要添加的边数是桥的个数/ ...

  10. poj 3101 Astronomy (java 分数的最小公倍数 gcd)

    题目链接 要用大数,看了别人的博客,用java写的. 题意:求n个运动周期不完全相同的天体在一条直线上的周期. 分析:两个星球周期为a,b.则相差半周的长度为a*b/(2*abs(a-b)),对于n个 ...