• 【C语言】内存操作,内存函数篇---memcpy,memmove,memset和memcmp内存函数的使用和模拟实现【图文详解】


    欢迎来CILMY23的博客喔,本篇为​【C语言】内存操作,内存函数篇---memcpy,memmove,memset和memcmp内存函数的使用和模拟实现【图文详解】,图文讲解四种内存函数,带大家更深刻理解C语言中内存函数的操作,感谢观看,支持的可以给个一键三连,点赞关注+收藏。 

    前言

    在结束上一期字符系列篇后,本篇我们将了解四种内存操作的函数,它们分别是memcpy,memmove,memset和memcmp。

    目录

    一、memcpy

    memcpy的介绍和使用 

     memcpy的模拟实现

    二、memmove

     memmove的介绍和使用

     memmove的模拟实现

    三、memset

    四、memcmp 


    一、memcpy

     memcpy可以在cplusplus网站查询,memcpy - C++ Reference (cplusplus.com)

    函数原型如下:

    void * memcpy ( void * destination, const void * source, size_t num );

    函数介绍如下:

     

    函数返回值和使用案例如下:

     

    memcpy的介绍和使用 

     memcpy是一个复制内存空间的函数,Copy block of memory,复制内存块,将 num 字节的值从指向的位置直接复制到目标指向的内存块。

    那具体是怎么使用的呢?

    我们来看一个整型数组的使用案例:

    1. #include
    2. #include
    3. int main()
    4. {
    5. int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    6. int arr2[60];
    7. memcpy(arr2, arr1, 20);
    8. for (int i = 0; i < 5; i++)
    9. {
    10. printf("%d", arr2[i]);
    11. }
    12. return 0;
    13. }

    结果如下:

     memcpy的模拟实现

    思路: 因为我们并不知道要接收什么样的数据类型,所以可以用void *来定义数据类型,因为void*不能直接加减整数,所以我们要将其转换成char * 的一个字节指针变量,然后进行加减,最后赋值给自己。

    1. void* my_memcpy(void* dest, const void* src, size_t num)
    2. {
    3. assert(dest && src);
    4. void* ret = dest;
    5. while (num--)
    6. {
    7. *(char*)dest = *(char*)src;
    8. dest = (char*)dest + 1;
    9. src = (char*)src + 1;
    10. }
    11. return ret;
    12. }

     写完后,我们看案例,如果目标空间和源空间重叠了呢?

    1. #include
    2. #include
    3. int main()
    4. {
    5. int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    6. int arr2[60];
    7. memcpy(arr1+3, arr1, 20);
    8. for (int i = 0; i < 5; i++)
    9. {
    10. printf("%d", arr1[i]);
    11. }
    12. return 0;
    13. }

    结果如下:

    我们用自己的拷贝呢?

    1. int main()
    2. {
    3. int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    4. my_memcpy(arr1+3, arr1, 20);
    5. for (int i = 0; i < 10; i++)
    6. {
    7. printf("%d ", arr1[i]);
    8. }
    9. return 0;
    10. }

    结果如下: 

     

    我们来看过程图:

    总结:

    1.memcpy的使用需要包括头文件string.h

    2.函数memcpy从source的位置开始向后复制num个字节的数据到destination指向的内存位置。
    3.memcpy函数在遇到  '\0' 的时候并不会停下来。
    4.如果source和destination有任何的重叠,复制的结果都是未定义的。

    5.memcpy的返回值是目标空间的起始地址

    二、memmove

    为了解决上述情况,我们需要用到memmove来解决内存重叠的问题,memmove可以在cplusplus网站查询,memmove - C++ Reference (cplusplus.com)

    函数原型如下:

    void * memmove ( void * destination, const void * source, size_t num );

    函数介绍如下:

    函数返回值和使用案例如下:

     memmove的介绍和使用

     memmove可以解决出现内存重叠空间的情况,将 num 字节的值从指向的位置复制到目标指向的内存块。复制就像使用中间缓冲区一样进行,从而允许目标重叠。

    1. int main()
    2. {
    3. int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    4. memmove(arr1+3, arr1, 20);
    5. for (int i = 0; i < 10; i++)
    6. {
    7. printf("%d ", arr1[i]);
    8. }
    9. return 0;
    10. }

    结果如下: 

     memmove的模拟实现

     思路:为了解决重叠问题,我们得对src和dest的目标空间起始位置进行讨论,分两种情况,从后往前或者从前往后复制

    我们发现当dest在src的右边就需要从后往前拷贝 

     

    而dest在src的左边就需要从前往后拷贝 

     

    当dest和src不重叠的时候,无论是从前往后,还是从后往前都可以。 

     所以一共有两种方案

    方案一,我们采取dest在src前面的情况,然后其余只采用从后往前

    方案二、我们采取dest >= src,并且,dest <= (char*)src+ num,

    1. //方案一
    2. void* my_memmove(void* dest, const void* src, size_t num)
    3. {
    4. assert(dest && src);
    5. void* ret = dest;
    6. if (dest < src)
    7. {
    8. //从前向后
    9. while (num--)
    10. {
    11. *(char*)dest = *(char*)src;
    12. dest = (char*)dest + 1;
    13. src = (char*)src + 1;
    14. }
    15. }
    16. else
    17. {
    18. //从后向前
    19. while (num--)
    20. {
    21. *((char*)dest + num) = *((char*)src + num);
    22. }
    23. }
    24. return ret;
    25. }
    1. //方案二
    2. void* my_memmove(void* dest, const void* src, size_t num)
    3. {
    4. assert(dest && src);
    5. void* ret = dest;
    6. if (dest > src && dest <= (char*)src +num)
    7. {
    8. //从后向前
    9. while (num--)
    10. {
    11. *((char*)dest + num) = *((char*)src + num);
    12. }
    13. }
    14. else
    15. {
    16. //从前向后
    17. while (num--)
    18. {
    19. *(char*)dest = *(char*)src;
    20. dest = (char*)dest + 1;
    21. src = (char*)src + 1;
    22. }
    23. }
    24. return ret;
    25. }

    二者最后的结果如下所示:

    总结:
    1.如果源空间和⽬标空间出现重叠,就得使用memmove函数处理。

    2.memmove的使用需要包括头文件string.h

    3.memmove将 num 字节的值将源指向的位置复制到目标指向的内存块。复制就像使用中间缓冲区一样进行,从而允许目标和源空间重叠。

    4.memmove和memcpy函数一样在遇到  '\0' 的时候并不会停下来。

    5.memmove的返回值是目标空间的起始地址

    三、memset

     memset可以在cplusplus网站查询,memset - C++ 参考 (cplusplus.com)

    函数原型如下:

    void * memset ( void * ptr, int value, size_t num );

    函数介绍如下:

     

    函数返回值和使用案例如下:

     

    函数的使用和介绍 

    memset是用来填充内存的,填充内存的值就是函数参数中的value

    1. #include
    2. #include
    3. int main()
    4. {
    5. char arr[] = "hello CILMY23";
    6. memset(arr, '1', 5);
    7. for (int i = 0; i < 13; i++)
    8. {
    9. printf("%c ", arr[i]);
    10. }
    11. return 0;
    12. }

     结果如下:

    总结:

    1.memset是用来设置内存的,将内存中的值以字节为单位设置成想要的内容。

    2.memset的使用需要包括头文件string.h

    3.memset的返回是原空间的地址

    4.memset只能对字节进行操作,操作的是一个字节数

    四、memcmp 

     memcmp可以在cplusplus网站查询,memcpy - C++ Reference (cplusplus.com)

    函数原型如下:

    int memcmp ( const void * ptr1, const void * ptr2, size_t num );

    函数介绍如下:

    函数返回值和使用案例如下:

     memcmp的使用

    1. #include
    2. #include
    3. int main()
    4. {
    5. int arr1[] = { 1,2,3,4,5 };
    6. int arr2[] = { 1,5,6,7,8, };
    7. int ret = memcmp(arr1, arr2,5);
    8. printf("%d ", ret);
    9. return 0;
    10. }

    总结:

    1.memcmp是比较从ptr1和ptr2指针指向的位置开始,向后的num个字节

    2.memcmp的使用需要包括头文件string.h

    3.memcmp的返回值是ptr1大于ptr2返回大于0的值,小于返回小于0的值,相等返回0

    4.memcmp比较的是字节内容

    感谢各位同伴的支持,本期内存函数篇就讲解到这啦,如果你觉得写的不错的话,可以给个一键三连,点赞关注+收藏,若有不足,欢迎各位在评论区讨论。   

  • 相关阅读:
    RAW-GNN: RAndom Walk Aggregation based Graph Neural Network
    内网渗透面试问题
    数据链路层概述
    SpringBoot 应用脚手架 Spring Initializr
    Vector Search和专用Search Nodes:现已正式发布
    秋招每日一题T3——设计循环双端队列
    [附源码]java毕业设计疫情居家隔离服务系统
    Win11无法删除文件夹怎么办?Win11无法删除文件夹的解决方法
    SringBoot快速实现参数校验
    π221N31 3.0kVrms 超低功耗 兼容I2C隔离双向通信 隔离器完美代替Si8602AB-AS
  • 原文地址:https://blog.csdn.net/sobercq/article/details/136245695