当前位置: 首页 > 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 函数是通过返回值告知⼤⼩关系的。

相关文章:

  • 空间复杂度** 与 **所需辅助空间**
  • LVGL源码学习之渲染、更新过程(1)---标记和激活
  • 我国脑机接口市场规模将破38亿元,医疗领域成关键突破口
  • SDC命令详解:使用all_inputs命令进行查询
  • 每天批次导入 100 万对账数据到 MySQL 时出现死锁
  • 一、对linux驱动文件编写时结构认识与记录
  • gpu硬件,gpu驱动,cuda,CUDA Toolkit,cudatoolkit,cudnn,nvcc概念解析
  • 操作系统 第2章节 进程,线程和作业
  • 【PhysUnits】3.3 SI 基础量纲单位(units/base.rs)
  • Leetcode刷题 由浅入深之字符串——541. 反转字符串Ⅱ
  • 振动临近失效状态,怎么频谱会是梳子?
  • API请求参数有哪些?
  • Modbus RTU 详解 + FreeMODBUS移植(附项目源码)
  • 【算法】:滑动窗口
  • 常见图像融合算法(alpha和金字塔融合)
  • 使用智能表格做FMEDA
  • Mysql--基础知识点--91.1--慢查询日志
  • 日常知识点之随手问题整理(思考单播,组播,广播哪个更省带宽)
  • RocketMQ 深度解析:架构设计与最佳实践
  • 学习黑客认识数字取证与事件响应(DFIR)
  • 教育部答澎湃:2025世界数字教育大会将发布系列重磅成果
  • 甘肃省政府原副省长赵金云被决定逮捕
  • 75岁亚当·费舍尔坐镇,再现80分钟马勒《第九交响曲》
  • 上海发布预付卡消费“10点提示”:警惕“甩锅闭店”套路
  • 上海优化营商环境再攻坚,企业和机构有哪些切实感受?
  • 首届上海老年学习课程展将在今年10月举办