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

【C/C++】C语⾔内存函数

C语言内存函数

1. memcpy使用和模拟实现

memcpy可以代替strcpy

void * memcpy ( void * destination, const void * source, size_t num );//void*来接受任意指针,size_t 单位是字节
//memcpy的头文件为<string.h>  mem是memory的缩写 是内存的意思

功能:

• memcpy 是完成内存块拷⻉的,不关注内存中存放的数据是啥**(遇到‘\0’不会停)**

• 函数 memcpy 从 source 的位置开始向后复制 num 个字节的数据到 destination 指向的内存位置。

• 如果 source 和 destination 有任何的重叠,复制的结果都是未定义的。

◦ (内存重叠的情况使⽤ memmove 就⾏)

• memcpy 的使⽤需要包含<string.h>

参数:

destination :指针,指向⽬标空间,拷⻉的数据存放在这⾥

source :指针,指向源空间,要拷⻉的数据从这⾥来

num :要拷⻉的数据占据的字节数

返回值:

拷⻉完成后,返回⽬标空间的起始地址

1.1 代码演示

#include <stdio.h>
#include <string.h>
int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };int arr2[10] = { 0 };memcpy(arr2, arr1, 20);//20字节int i = 0;for (i = 0; i < 10; i++){printf("%d ", arr2[i]);}return 0;
}

1.2 模拟实现

void * memcpy ( void * dst, const void * src, size_t count)//dst要变 src不变
{void * ret = dst;//记录初始位置assert(dst);assert(src);/** copy from lower addresses to higher addresses*/while (count--) {*(char *)dst = *(char *)src;//强转成char* 一次换一个字节dst = (char *)dst + 1;//(char* dst)++src = (char *)src + 1;//(char* src)++}return(ret);
}

那这时候就有人问了?

有重叠的 如 memcpy(arr1+2,arr1,20) 会发生什么情况

在这里插入图片描述

3和4被1 2替换后 5会被红色的1替代 从而达不到想要的效果

所以来看接下来函数

2. memmove使用和模拟实现

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

功能:

• memmove函数也是完成内存块拷⻉的

• 和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。

• memmove的使⽤需要包含<string.h>

参数:

destination :指针,指向⽬标空间,拷⻉的数据存放在这⾥

source :指针,指向源空间,要拷⻉的数据从这⾥来

num :要拷⻉的数据占据的字节数

返回值:

拷⻉完成后,返回⽬标空间的起始地址

2.1 代码演示

#include <stdio.h>
#include <string.h>
int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };memmove(arr1 + 2, arr1, 20);//arr1+2就是数组中3的位置 20=4*sizeof(int)int i = 0;for (i = 0; i < 10; i++){printf("%d ", arr1[i]);}return 0;
}

输出的结果:

1 2 1 2 3 4 5 8 9 10

2.2 模拟实现

在这里插入图片描述

此种情况可以先把5移动到7 然后4移动到6 也就是从后往前

在这里插入图片描述

此种情况可以把先把3移动到1处 以此类推

我们可以得出结论 当dest在src左边时 是从前往后

当dest在src右边时 是从后往前

不重叠是都可以

void* memmove(void* dst, const void* src, size_t count) 
{void* ret = dst;//要返回起始空间if (dst < src) {/** Non-Overlapping Buffers* copy from lower addresses to higher addresses*/while (count--) {*(char*)dst = *(char*)src;dst = (char*)dst + 1;src = (char*)src + 1;}}else {/** Overlapping Buffers* copy from higher addresses to lower addresses*/dst = (char*)dst + count - 1;src = (char*)src + count - 1;while (count--) {*(char*)dst = *(char*)src;dst = (char*)dst - 1;src = (char*)src - 1;}}return(ret);
}
void* memmove(void* dst, const void* src, size_t count) 
{void* ret = dst;//要返回起始空间if (dst < src) {/** Non-Overlapping Buffers* copy from lower addresses to higher addresses*/while (count--) {*(char*)dst = *(char*)src;dst = (char*)dst + 1;src = (char*)src + 1;}}else {while(count--)//count=20{((char*)dest+count)=*((char*)src+count);//count=19}}}return(ret);
}

3. memset函数的使用

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

功能:

• memset 函数是⽤来设置内存块的内容的,将内存中指定⻓度的空间设置为特定的内容。

• memset 的使⽤需要包含 <string.h>

参数:

ptr :指针,指向要设置的内存空间,也就是存放了要设置的内存空间的起始地址。

value :要设置的值,函数将会把 value 值转换成 unsigned char 的数据进⾏设置的。也就是以字节为单位来设置内存块的。

num :要设置的内存⻓度,单位是字节。

**返回值:**返回的是要设置的内存空间的起始地址。

3.1 代码演示

#include <stdio.h>
#include <string.h>
int main ()
{char str[] = "hello world";memset (str,'x',6);//以字节为单位printf(str);return 0;
}

输出的结果:

xxxxxxworld

3.2 总结:

当有⼀块内存空间需要设置内容的时候,就可以使⽤memset函数,值得注意的是memset函数对内存 单元的设置是以字节为单位的。

4. memcmp函数的使用

内存块比较

int memcmp ( const void * ptr1, const void * ptr2, size_t num );//因为是void* 所以任何类型都能比较 strcmp只能比较字符串 

功能:

⽐较指定的两块内存块的内容,⽐较从ptr1和ptr2指针指向的位置开始,向后的num个字节 memcmp 的使⽤需要包含 <string.h>

参数:

ptr1 :指针,指向⼀块待⽐较的内存块

ptr2 :指针,指向另外⼀块待⽐较的内存块

num :指定的⽐较⻓度,单位是字节

返回值:

在这里插入图片描述

4.1 代码演示

#include <stdio.h>
#include <string.h>
int main()
{int arr1[] = { 1,2,3,4,5,6,7 };int arr2[] = { 1,2,3,4,8,8,8 };int n = memcmp(arr1, arr2, 17);//比较17个字节 vs中是大端储存 比较的时候到了第17字节 arr1是05 arr2是08printf("%d\n", n);return 0;
}

4.2 总结

如果要⽐较2块内存单元的数据的⼤⼩,可以使⽤ memcmp 函数,这个函数的特点就是可以指定⽐较 ⻓度。 memcmp 函数是通过返回值告知⼤⼩关系的。

http://www.xdnf.cn/news/353377.html

相关文章:

  • [Errno 122] Disk quota exceeded
  • Linux59 SSH配置前瞻 JumpServer双网卡ping通
  • 金仓数据库永久增量备份技术原理与操作
  • 电商平台如何做好DDoS 攻防战?
  • 物流基础知识-术语 | 医药物流(1)
  • OpenHarmony平台驱动开发(十),MMC
  • k8s监控方案实践(二):集成Alertmanager告警与钉钉Webhook通知
  • C23 与 MISRA C:2025:嵌入式 C 语言的进化之路
  • 4.3【LLaMA-Factory实战】教育大模型:个性化学习路径生成系统全解析
  • 微服务中 本地启动 springboot 无法找到nacos配置 启动报错
  • 第十六章,网络型攻击防范技术
  • Python 常用内置函数详解(十):help()函数——查看对象的帮助信息
  • 【论文阅读27】-TCN–BiLSTM -滑坡预测
  • 从Dockerfile 构建docker镜像——保姆级教程
  • `待办事项css样式
  • electron 结合 react(cra创建的) 创建桌面应用和打包桌面应用
  • 2025年API安全防御全解析:应对DDoS与CC攻击的智能策略
  • rtsp,。。。。
  • 现代框架对SEO的深度影响
  • 【概念解读】开发中遇到的概念阶段
  • RuntimeError: expected scalar type ComplexDouble but found Float
  • 计算机视觉与深度学习 | 视觉+激光雷达+惯惯性SLAM算法汇总(原理,公式,代码)
  • Java中的分布式缓存与Memcached集成实战
  • 电压取样端口静电浪涌防护方案 之6TS Series瞬态抑制器TVS
  • hz2新建Keyword页面
  • 使用 swift 微调 Qwen3-4b 模型
  • 矩阵短剧系统:如何用1个后台管理100+小程序?深度解析多端绑定技术
  • C++--类中this指针的讲解
  • 从数据孤岛到智能工厂:RG3000边缘网关的数字化转型实践
  • Mac QT水平布局和垂直布局