C语言----C语言内存函数

embedded/2024/9/23 2:25:56/

1.memcpy--内存拷贝--使用和模拟实现 

//memcpy基本格式:
//               目标空间地址       原空间地址        被拷贝的字节个数
//void *memcpy(void * destination, const void * source,size_t num);
//因为内存拷贝拷贝的数据有:整型数据、结构体数据、结构体数据。那么我们就要用到void*进行接收任意类型的数据
//void*指针可以接收任意空间//size_t num这个是要被拷贝的字节个数
////返回的是目标空间的起始地址int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };int arr2[20] = { 0 };memcpy(arr2, arr1 + 2, 20);//因为我们要拷贝的是3~7,那么就是5个整型,20个字节//因为我们要从3开始进行拷贝//恰好因为数组名是首元素的地址,那么3的地址就是arr+2//将arr1中的3 4 5 6 7拷贝放到arr2中for (int i = 0; i < 20; i++){printf("%d ", arr2[i]);}return 0;
}
//我们要满足目标空间能进行修改
//目标空间足够大//这个memcpy函数是不会考虑\0的,我让他拷贝多少个字节他就拷贝多少个字节

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

1.这个函数在遇到\0的时候并不会停下来

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

memcpy函数最终返回的是目标空间的起始地址

//函数的一种写法:
这个函数最终返回的是目标空间的起始地址
//void* my_memcpy(void *dest, const void*src, size_t num)//目标空间是要进行修改的,但是源头是不能被修改的,所以我们加上const
//{
//    assert(src && dest);//防止两个指针是空指针
//    void* ret = dest;//我们需要先创建一个指针存放最开始的dest,通过ret我们可以找到dest指向的数组
//    //这里的src指向的是数组中3的位置
//    //dest指向的是arr2数组的首元素的位置
//    //拷贝20个字节
//    //void*的指针不能进行直接计算,void*的指针是无具体类型的指针
//    for (int i = 0; i < num; i++)//一对字节一对字节
//    {
//        *(char*)dest = *(char*)src;//强制类型转换再进行解引用就将那一对字节进行交换
//        dest = (char*)dest + 1;//先将dest强制类型转换,在进行+1操作,因为上面的强制类型转换只是暂时的,为了能让dest进行++操作,我们就这样操作
//        src = (char*)src + 1;
//    }
//    return ret;
//}//函数的另一种写法:
//函数的一种写法:
//这个函数最终返回的是目标空间的起始地址
void* my_memcpy(void *dest, const void*src, size_t num)
{assert(src && dest);void* ret = dest;while (num--)//也是循环num次{*(char*)dest = *(char*)src;dest = (char*)dest + 1;src = (char*)src + 1;}return ret;
}int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };int arr2[20] = { 0 };int *p=my_memcpy(arr2, arr1 + 2, 20);//这个函数的作用就是我们在arr1中选择的那部分粘贴在arr2中for (int i = 0; i < 20; i++){printf("%d ", p[i]);}return 0;
}//memcpy函数不负责重叠内存的拷贝,非要使用,结果就是未定义的//只负责不重叠的内存

函数的返回值是void*类型的数据

这个memcpy函数有三个数据

2.memmove--内存移动--使用和模拟实现 

2.memmove--内存移动--使用和模拟实现

int main()
{//将123456拷贝放到34567上面int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };memmove(arr1 + 2, arr1, 20);for (int i = 0; i < 10; i++){printf("%d ", arr1[i]);}//输出结果是1 2 1 2 3 4 5 8 9 10return 0;
}
//memmove就是用来处理这些重叠内存的拷贝
void* my_memmove(void *dest, const void*src, size_t num)
{void* ret = dest;//将目标空间起始地址记录下来assert(dest && src);//保证指针的有效性//最开始的dest指向的是3,src指向的是1,字节是20if (dest < src)//前-->后//目标空间的首元素<拷贝的起始元素--就相当与我们把34567拷贝到12345//那么12的位置就是dest<src的位置{while (num--)//总共拷贝num个字节,那么就是num--,等所有字节拷贝完循环就停止了{*(char*)dest = *(char*)src;//拷贝完一对字节就往后走,进行+1操作dest = (char*)dest + 1;src = (char*)src + 1;}}else//后-->前剩下的区域---dest > src{while (num--){//进入循环的时候,num 已经是19了//*((char*)dest + num)---找到最后一个字节的位置*((char*)dest + num)    =       *((char*)src + num);//目标的最后一个位置的字节------    起源空间的最后一个位置//将起源的最后一个字节先开始挪动,从最后一个位置依次挪动//就是将我们要挪动的那块区域,我们从最右边开始挪动}}return ret;
}
int main()
{//将123456拷贝放到34567上面int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };my_memmove(arr1 + 2, arr1, 20);for (int i = 0; i < 10; i++){printf("%d ", arr1[i]);//1 2 1 2 3 4 5 8 9 10}return 0;
}//总之:这个拷贝是分三块区域的,最前面的一块区域满足dest<src
//我们只能从前往后进行拷贝,不然会出错误//而剩下的两块区域可以同时从后往前进行拷贝,那么我们就将这两块区域放在一起

在C语言标准中,明确规定了memcpy只要能实现不重叠的拷贝就行,重叠的拷贝交给memmove

我们发现vs上面的库函数memcpy函数也能实现重叠内存的拷贝

我们在以后的拷贝中,我们可以用memmove,因为不管是重叠的还是不重叠的都能搞定

3.memset--内存设置--函数的使用 

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

基本格式:

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

ptr就是指向要被填充的内存块的指针

value就是我们想要设置的值是什么

num是设置的字节的大小

int main()
{char arr[] = "hello world";//将world改成5个xmemset(arr + 6, 'x', 5);//数组首元素的地址+6就是指向的是wprintf("%s\n", arr);return 0;
}

4.memcmp--内存比较--函数的使用 

4.memcmp--内存比较--函数的使用

内存的比较

什么类型的都能进行比较

int main()
{int arr1[] = { 1,2,3,4,5 };int arr2[] = { 1,2,3,6,5 };int ret=memcmp(arr1, arr2, 16);printf("%d\n", ret);//输出的数据就是-1//就是说明arr1<arr2return 0;
}

http://www.ppmy.cn/embedded/50937.html

相关文章

LDR6500:手机电脑拓展坞转接器方案的卓越之选

随着科技的飞速发展&#xff0c;手机和电脑已成为我们日常生活中不可或缺的工具。然而&#xff0c;它们的接口有限&#xff0c;经常难以满足我们多样化的需求。这时&#xff0c;一款高效、稳定的拓展坞转接器就显得尤为重要。LDR6500&#xff0c;作为乐得瑞科技精心研发的USB P…

一站式家装服务管理系统的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;管理员管理&#xff0c;装修风格管理&#xff0c;主材管理&#xff0c;用户管理&#xff0c;基础数据管理 前台账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;装修风格&#xff0…

前端菜鸡学习日记 -- 关于pnpm

哈咯哇大家&#xff0c;我又来了&#xff0c;最近稍微悠闲一些&#xff0c;所以就趁着这个机会学习一些新的知识&#xff0c;今天就是碰巧遇到了pnm&#xff0c;这个可以看作是npm的升级版本&#xff0c;比npm要快&#xff0c;用起来也更得劲更迅速 官网地址&#xff1a;https…

USB学习——12、usb初始化和插拔驱动软件流程大致框架描述

usb初始化和插拔驱动软件流程大致框架描述&#xff1a; 当设备启动时&#xff0c;usb的主机控制器设备驱动&#xff08;HCD&#xff09;和 usb的root hub会先初始化&#xff1a; 1、xhci-plat.c主机控制器驱动那里&#xff0c;__usb_creat_hcd创建usb主机数据结构&#xff0c;m…

SNMP学习

文章目录 前言基本介绍端口和网络层特性工作原理应用场景版本总结 前言 SNMP&#xff08;Simple Network Management Protocol&#xff0c;简单网络管理协议&#xff09;是一种应用层协议&#xff0c;用于网络管理&#xff0c;允许网络管理员监控和管理网络设备的状态和配置。…

jeecg快速启动(附带本地运行可用版本下载)

版本整理&#xff08;windows x64位&#xff09;&#xff1a; redis&#xff1a;3.0.504 MYSQL&#xff1a;5.7 Maven&#xff1a;3.9.4(setting文件可下载) Nodejs&#xff1a;v16.20.2&#xff08;建议不要安装默认路径下&#xff0c;如已安装在c盘&#xff0c;运行yarn报…

AJAX中get和post的区别

在AJAX&#xff08;Asynchronous JavaScript and XML&#xff09;中&#xff0c;GET 和 POST 是两种常用的HTTP请求方法&#xff0c;它们之间存在一些关键的区别。以下是这些区别的主要点&#xff1a; 请求的目的&#xff1a; GET&#xff1a;通常用于从服务器检索&#xff08;…

涉密文件当废品卖,涉密文件如何安全便捷销毁?

前几天&#xff0c;一位大爷在废品收购站买到四本涉及军事的涉密文件登上热搜&#xff0c;此事源于相关工作人员没有按照涉密文件销毁流程&#xff0c;缺乏保密意识&#xff0c;将200余本涉密资料当做废品出售&#xff0c;导致涉密信息在外部曝光。 无论是在企业内部还是在机关…