当前位置: 首页 > news >正文

C语言内存函数(21)

文章目录

  • 前言
  • 一、memcpy的使用和模拟实现
  • 二、memmove的使用和模拟实现
  • 三、memset函数的使用
  • 四、memcmp函数的使用
  • 总结


前言

  正文开始,发车!


一、memcpy的使用和模拟实现

函数模型:void* memcpy(void* destination, const void* source, size_t num);

使用注意事项:
1.函数 memcpy 从 source 的位置开始向后复制 num 个字节的数据到 destination 指向的内存位置
2.这个函数在遇到 ‘\0’ 的时候并不会停下来,给多少就复制多少
3.如果 source 和 destination 有任何的重叠,复制的结果都是未定义

// 使用举例
int main()  
{  int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };  int arr2[10] = {0};  //将arr1中的1 2 3 4 5,拷贝到arr2中  memcpy(arr2, arr1, 5*sizeof(int));  int i = 0;  for (i = 0; i < 10; i++)  {  printf("%d ", arr2[i]);  // 1 2 3 4 5 0 0 0 0 0}  return 0;  
}

我们来模拟实现一下 my_memcpy ,首先所接收的 dest 、src 都是不明确类型的,用 void* 接收,返回类型为 void*,且 src 并不希望被修改,可以加个 const 修饰

其实没那么复杂,我们有 num 个字节需要复制,那么直接循环 num 次就可以了,只是需要把 dest 和 src 强制转化成 char* 的类型就OK了
在这里插入图片描述

标红是因为什么原因?

原来,强制类型转化是临时的,下面两句各自加上就好了

dest = (char*)dest + 1;
src = (char*)src + 1;

这时候,我们再来回想一下为什么尽量要避免 dest 和 src 两者发生重叠
你脑海里想象一下 my_memcpy(arr1 + 2, arr1 + 1, 5 * sizeof(int)); 的过程,就明白了

下面是memcpy的完整模拟实现:

void* my_memcpy (void * dst, const void * src, size_t count )  
{  void * ret = dst;  assert(dst);  assert(src);  /*  * copy from lower addresses to higher addresses  */  while (count--) {  *(char *)dst = *(char *)src;  dst = (char *)dst + 1;  src = (char *)src + 1;  }  return(ret);  
}

二、memmove的使用和模拟实现

那怎么处理内存重叠的问题呢,那就用这个!

函数原型:void* memmove(void* destination, const void* source, size_t num);

使用注意事项:
1.可以把 memmove 当作可以处理重叠的 memcpy 吗? 可以!

我们来看看这个 memmove 为什么可以解决内存重叠情况下的复制:

当 src 在 dest 左边且发生重叠的时候,这时候如果从左往右复制,dest 所指向的 3 立马就被覆盖,等到 src 来复制的时候,已经只能复制 1 了,这不是我们想要的,于是,我们考虑从右向左复制,也就是 dest 的 7 被 5 覆盖 , 6 被 4 覆盖 … 3 被 1 覆盖
在这里插入图片描述

当 src 在 dest 右边且发生重叠的时候,这时候从右往左复制,又会发生上述的提前覆盖的情况,解决方法是什么?从左向右复制!
在这里插入图片描述

当 src 与 dest 不发生重叠的时候,从左向右 或者 从右向左 复制都没影响,所以我们想出一个总的复制方案

当 src < dest 的时候,从右向左复制
当 dest < src 的时候,从左向右复制

我们来思考一下 从右往左 复制该怎么实现,首先 dest 和 src 先跳到未部,注意要减一个1

dest = (char*)dest + num - 1;
src = (char*)src + num - 1;

接着开始往回复制,这与从左向右几乎等同,无非就是自加变为自减,所以,完整的 my_memmove 如下

void* my_memmove(void* dest, const void* src, size_t num)
{void* ret = dest;if (dest < src) {// 从左往右while (num--) {*(char*)dest = *(char*)src;dest = (char*)dest + 1;src = (char*)src + 1;}}else {// 从右往左dest = (char*)dest + num - 1;src = (char*)src + num - 1;while (num--) {*(char*)dest = *(char*)src;dest = (char*)dest - 1;src = (char*)src - 1;}}return ret;
}

三、memset函数的使用

函数原型:void* memset(void* ptr, int value, size_t num);

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

// 使用实例
#include <stdio.h>  
#include <string.h>  
int main ()  
{  char str[] = "hello world";  memset (str,'x',6);  printf(str);  // xxxxxxworldreturn 0;  
}

请注意!memset在设置的时候,是以字节为单位来设置的

// 错误使用举例
int main()  
{  int arr[10] = { 0 };  memset(arr, 1, 40);  //errreturn 0;  
}

因为它是以字节为单位来设置的,所以你可以想象,一个 int 有四个字节,每个字节都是1
也就是说,上述数组 arr 的每个元素都是 0x01010101,而不是我们想要的0x00000001

我们一般拿来清空、初始化数组或者结构体

四、memcmp函数的使用

函数原型:int memcmp(const void* ptr1, const void* ptr2, size_t num);

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

// 使用实例
int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };int arr2[] = { 1,2,3,4,8 };int ret = memcmp(arr1, arr2, 16); // 0printf("%d\n", ret);ret = memcmp(arr1, arr2, 17); // -1printf("%d\n", ret);return 0;
}

至于原理,请调试并打开内存查看,就很清楚了
在这里插入图片描述


总结

我们在学习的时候应该带有自己的思考,比如上述 arr1 和 arr2 的图片可能有一个地方会引起你的注意

对于每个 int 的四个字节,数据低位同时也是存放在地址的低位
举例来说,假如把 int a = 1;存放在内存里面:
0x00000093F16FF6A8 :存放01
0x00000093F16FF6A9 :存放00
0x00000093F16FF6AA :存放00
0x00000093F16FF6AB :存放00

你可能跟我一样,对数据在内存中的存储由此产生了极大的疑问和兴趣,没关系,我们下篇文章开始介绍

http://www.lryc.cn/news/437143.html

相关文章:

  • 三高基本概念之-并发和并行
  • 宝塔面板FTP连接时“服务器发回了不可路由的地址。使用服务器地址代替。”
  • 面试的一些小小经验
  • IV转换放大器原理图及PCB设计分析
  • 【数学建模经验贴】一个研赛数模老手的经验
  • vivo手机已删除的短信还能恢复吗?
  • [网络][CISCO]CISCO IOS升级
  • 通过python提取PDF文件指定页的图片
  • Leetcode Hot 100刷题记录 -Day12(轮转数组)
  • GitHub每日最火火火项目(9.13)
  • 力扣--649.Dota2参议院
  • vim 安装与配置教程(详细教程)
  • 【WPF】Popup的使用
  • 力扣刷题之2576.求出最多标记下标
  • 黑马JavaWeb开发笔记16——请求(postman、简单参数、实体参数、@RequestParam映射)
  • Corrupt block relative dba: 0x02c0b382 (file 11, block 45954)
  • 二叉排序树在实际生活应用中作用
  • 单例模式的学习
  • 54 mysql 中各种 timeout - connect/wait/interactive/read/write_timeout
  • 实战案例(5)防火墙通过跨三层MAC识别功能控制三层核心下面的终端
  • 【智能流体力学】数值模拟中的稳态和瞬态
  • Vue-Route4 ts
  • sizeof和strlen的小知识
  • Java项目: 基于SpringBoot+mybatis+maven宠物咖啡馆平台(含源码+数据库+毕业论文)
  • 戴尔14代服务器配置IDRAC9远程配置说明
  • 如何让你家里的电脑连接公司的远程桌面
  • 软件:分享8个常用视频剪辑免费软件,你都用过吗?
  • TS 常用类型
  • 半导体芯闻--20240913
  • C盘空间不足如何解决?解决C盘空间不足的7个方法