memset、memcpy、memmove的常见用法和区别,以及memcpy与strcpy的区别

今天听到有个同学,去面试。考到了关于结构体之间内容的复制,题目大概是知道两个结构体名A,B,但不知道A,B结构体中的内容。要求你把A结构体的成员,复制到B结构体中。

在这里需要用到memcpy函数。所以我今天把相关mem函数进行一个小结。供大家参考。有什么不对的地方,请指出。


strcpy


原型:extern char *strcpy(char *dest,char *src); 

用法:#include <string.h>

功能:把src所指由NULL结束的字符串复制到dest所指的数组中。

说明:src和dest所指内存区域不可以重叠dest必须有足够的空间来容纳src的字符串。

返回值:指向dest的指针。

例:char a[100],b[50];strcpy(a,b);

如用strcpy(b,a),要注意a中的字符串长度(第一个‘\0’之前)是否超过50位,如超过,则会造成b的内存地址溢出。


memcpy 


原型:extern void *memcpy(void *dest, void *src, unsigned int count);

用法:#include <string.h>

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

说明:src和dest所指内存区域不能重叠

函数返回值:指向dest的指针。可以拿它拷贝任何数据类型的对象。

举例:char a[100],b[50]; memcpy(b, a, sizeof(b));

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


memset


原型:extern void *memset(void *buffer, int c, int count);

用法:#include <string.h>

功能:把buffer所指内存区域的前count个字节设置成字符c。

说明:返回指向buffer的指针。用来对一段内存空间全部设置为某个字符。

举例:char a[100];memset(a, '\0', sizeof(a));

memset可以方便的清空一个结构类型的变量或数组。

用memset就非常方便:

memset(&stTest,0,sizeof(struct sample_struct));

如果是数组:
struct sample_struct    TEST[10];

则memset(TEST,0,sizeof(struct sample_struct)*10);


memmove


原型:void *memmove( void* dest, const void* src, size_tcount );

头文件:<string.h>

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

说明:memmove用于从src拷贝count个字符到dest,如果目标区域和源区域有重叠的话,memmove能够保证源串在被覆盖之前将重叠区域的字节拷贝到目标区域中。但复制后dest内容会被更改。但是当目标区域与源区域没有重叠则和memcpy函数功能相同。函数返回指向dest的指针

程序示例

// memmove.c

#include <stdio.h>

#include <string.h>

int main(void)

{

char s[]="Golden Global View";

memmove(s,s+7,strlen(s)+1-7);

printf("%s",s);

getchar();

return 0;

}


Memcpy与strcpy的实现和区别:


mem是对内存,str是一段字符串进行操作,不过它的长度是以‘\0’来做最后一位,随时都可以计算出来所以memcpy需要第三个参数,而strcpy不需要。

void* memcpy(void* pvTo, const void* pvForm, size_t size)
{
assert((pvTo!= NULL) && (pvFrom!= NULL));//使用断言防止传递空地址
unsigned char* pbTo = (unsigned char*)pvTo;//防止改变pvTo的地址
unsigned char* pbFrom = (unsigned char*)pvFrom;//防止改变pvFrom的地址
while(size-- > 0)
{
   *pbTo++ = *pbFrom++;
}
return pvTo;
}

char* strcpy(char* pDest, const char* pSrc)
{
assert((pDest != NULL) && (pSrc != NULL));
char *pTmp = pDest;
while ((*pDest++ = *pSrc++) != '\0')
   ;
return pTmp;
}


memcpy和memmove 的用法和区别:


void *memmove(void *dest, const void *source, size_t count)
{
//    assert((NULL != dest) && (NULL != source));  //如果它的条件返回错误,则终止程序执行

                                           详解见文章结尾小知识点补充。
    char *tmp_source = (char *)source;
    char *tmp_dest = (char *)dest;
    
    if(tmp_dest <= tmp_source || (tmp_source + count) <= tmp_dest)
    {                                  // 如果没有重叠区域
        while(count--)
            *tmp_dest++ = *tmp_source++;
    }
    else
    {                                  //如果有重叠
        tmp_source += count - 1;
        tmp_dest += count - 1;
        while(count--)
            *tmp_dest-- = *tmp_source--;
    }
    return dest;
}

    函数memcpy()从source指向的区域向dest指向的区域复制count个字符,如果两数组重叠,不定义该函数的行为.而memmove(),如果两函数重叠,赋值仍正确进行。 
    memcpy函数假设要复制的内存区域不存在重叠,如果你能确保你进行复制操作的的内存区域没有任何重叠,可以直接用memcpy;如果你不能保证是否有重叠,为了确保复制的正确性,你必须用memmove。


【补充知识点】

assert() 宏用法

注意:assert是宏,而不是函数。在C的assert.h 头文件中。

assert宏的原型定义在<assert.h>中,其作用是如果它的条件返回错误,则终止程序执行,原型定义:

#include <assert.h>

void assert( int expression );

assert的作用是先计算表达式 expression ,如果其值为假(即为0),那么它先向标准错误流stderr打印一条出错信息,然后通过调用 abort 来终止程序运行;否则,assert()无任何作用。宏assert()一般用于确认程序的正常操作,其中表达式构造无错时才为真值。完成调试后,不必从源代码中删除assert()语句,因为宏NDEBUG有定义时,宏assert()的定义为空。

这篇文章里的内容就是我对memset、memcpy、memmove的常见用法和区别,以及memcpy与strcpy的区别的理解。我有什么遗漏吗?请指正。 不论你的水平如何,请留下你的想法。


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
引用: 结果发现:memcpy在拷贝数据时与strcpy和strncpy不同的是memcpy遇到 '\0' 是不会停止拷贝到。 引用: 知识点4:memcpy()函数的模拟:void* my_memcpy(void* dst, const void* src, size_t count) { ... } 引用: 代码演示:memcpy(arr2, arr1, 6); memsetmemcpy是C语言中的两个函数。memset函数用于将一块内存区域的每个字节都设置为指定的值,而memcpy函数用于将一块内存区域的数据拷贝到另一块内存区域。 具体来说,memset函数的原型为void *memset(void *s, int c, size_t n),其中s是指向内存区域的指针,c是要设置的值,n是要设置的字节数。该函数会将s所指向的内存区域的每个字节都设置为c。 而memcpy函数的原型为void *memcpy(void *dest, const void *src, size_t n),其中dest是目标内存区域的指针,src是源内存区域的指针,n是要拷贝的字节数。该函数会将src所指向的内存区域的数据拷贝到dest所指向的内存区域。 需要注意的是,memcpy在拷贝数据时,不会像strcpy和strncpy那样遇到 '\0' 就停止拷贝,而是会一直拷贝下去直到拷贝完指定的字节数。这是memcpystrcpy和strncpy的一个区别。 在引用中给出了一个自定义的my_memcpy函数的示例代码,该函数模拟了memcpy函数的功能。 在引用中给出了一个使用memcpy函数的示例代码,该代码将字符串"abc\0def"的前6个字节拷贝到另一个字符数组中。 综上所述,memsetmemcpy是C语言中用于操作内存的两个函数,分别用于设置内存区域的值和拷贝内存区域的数据。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [内存函数:memcpymemmove、memcmp、memset(超详细讲解,小白一看就懂!!!!)](https://blog.csdn.net/weixin_45031801/article/details/127481057)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值