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

C语言---内存函数

memcpy函数的使用及模拟实现

        memcpy的功能和strcpy类似,都是用来拷贝数据的。与strcpy不同的是,memcpy的适用性更广并且是以字节为单位来拷贝的。

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

        memcpy函数的作用就是拷贝从source这个地址开始向后num个字节的数据destinction里边去。返回的是的destinction的首地址。

        使用案例 

        模拟实现

                分析:

                在上边知道了memcpy的使用方式以及形参和返回值之后,我们可以试着模拟实现一下。

                由于传过来的参数都是void*类型的指针,而我又不知道我将要处理什么类型的数据,恰恰我们知道所要处理的字节数,所以我们想将其强转为char*类型来以字节为单位处理数据。其本质也是拷贝数据,拷贝的次数就是num次,一个一个字节的拷贝,用循环来实现。

                代码实现:

#include<stdio.h>
#include<assert.h>
void* my_memcpy(void* des, const void* sor, size_t num)
{assert(des && sor);void* p = des;while (num--){*(char*)des = *(char*)sor;//先强转再++((char*)des)++;((char*)sor)++;}return p;
}int main()
{int arr1[] = { 1,2,3,4,5 };int arr2[20] = { 0 };my_memcpy(arr2, arr1, 20);return 0;
}

memmove函数的使用及模拟实现

        其实memmove和memcpy函数的功能是一样的,两者其实是用于不同的场景,memmove用于拷贝的数据是可以重叠的。

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

    memmove函数的形参跟返回值的意义和memcpy一模一样

    何谓可以重叠?

    假设还是那个arr1数组,里边还是存放着1,2,3,4,5。但我现在不将它拷贝到arr2数组里边去了,我想将arr1数组的内容拷贝到arr1+1的位置,还是拷贝20个字节。这个时候用memcpy函数就显得有点力不从心了,memmove函数就是可以解决这个问题。

        使用案例

        模拟实现 

                分析:

                由于情况较多,我用画图的形式来呈现。                

                从图片里边的两种情况来看,我们在模拟实现memmove的时候,就是要考虑到这些点,

简单总结一下,当sor在des所在矩形的里边的时候,我们是需要从前往后拷贝数据的,此时的代码逻辑跟刚才的memcpy是一样的。而当sor在des所在矩形左边的时候,就要从后往前拷贝,此时就需要我们先找到要拷贝的最后一个元素的地址,从上图来看,就是我们要找到5末尾的地址和7末尾的地址来从后往前拷贝。

                那具体怎么找呢?

                分析完了以上的信息,还有一种情形3 (这种情况从后往前和从前往后都不影响)

                但为了代码的方便,我们选择从后往前的方式。      

                代码实现:

#include<stdio.h>
#include<assert.h>
void* my_memmove(void* des,const void* sor,size_t num)
{assert(des && sor);void* ret = des;//当sor在des的左边的时候,就从后往前实现if (sor < des){//先找到要拷贝的最后一个数据的末尾的地址,然后实现从后往前交换//假设我要拷贝五个int类型的数据,那一开始num是20,num--之后就会变成19//而des和sor指针经过类型转换之后再加上num就能得到我们想要的地址//大家可以对照前边图片里边的 char arr2[20] 数组while (num--){*((char*)des + num) = *((char*)sor + num);}}//接下来就是从前往后拷贝,实现逻辑跟memcpy一样else{while (num--){*(char*)des = *(char*)sor;//先强转再++((char*)des)++;((char*)sor)++;}}return ret;
}int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };my_memmove(arr1 + 1, arr1, 20);return 0;
}

memset的使用

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

        memset是用于以字节为单位来修改数据的,一般用于字符数组和整型数组设置其值全部为0的时候。修改字符数组这很好理解,因为字符数组里边的值都是只占一个字节,以字节修改是最合适的。但为什么整型数组除了0,其他的值修改不了呢?我会在代码里边用注释来解释)

        解释一下形参和返回值:

        value是你想要修改的值,num是你想要修改的字节个数,ptr是你想要修改的起始地址,返回的也是也是该起始地址。

        使用案例1:

        使用案例2: 

        使用案例3:

        对于使用案例3,出现这样结果的原因就是因为memset是按照字节来修改数据的。

   对于一个int类型的数据,它是有4个字节

   对于1来说,它的补码就是

   00000000 00000000 00000000 00000001

   那要按照字节来修改为1的话,就变成了

   00000001 00000001 00000001 00000001

   而这个就是一个非常大的数字

memcmp的使用

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

        memcmp是用来比较从ptr1和ptr2指向的内存的数据进行比较。从ptr1和ptr2开始,向后比较num个字节。它的返回值跟strcmp类似。

        如果ptr1>ptr2,就返回大于0的数字,如果小于,就返回小于0的数字,如果等于,就返回等于0的数字。

        一旦遇到不相等的字节,它就不会往下边去比较了,直接返回。

        使用案例:

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

相关文章:

  • Axure通过下拉框选项改变,控制字段显隐藏
  • Rust 学习笔记:关于泛型的练习题
  • Switch最新 模拟器 Eden(伊甸)正式发布 替代Yuzu模拟器
  • C#面:Server.UrlEncode、HttpUtility.UrlDecode的区别
  • Python里字典的操作
  • C#语法篇 :基类子类转换,成员变化情况
  • 云蝠智能大模型呼叫动态情感共情能力上线!
  • SIGIR25-推荐论文整理
  • 面试相关的知识点
  • vue3 + vite 使用tailwindcss
  • 现代化SQLite的构建之旅——解析开源项目Limbo
  • 第17天-Pandas使用示例
  • 【SPIN】PROMELA 通道(Channels)(SPIN学习系列--8)
  • 【完整版】基于laravel开发的开源交易所源码|BTC交易所/ETH交易所/交易所/交易平台/撮合交易引擎
  • 机器学习-KNN算法
  • 为什么服务器突然变慢?从硬件到软件的排查方法
  • 论文阅读:Next-Generation Database Interfaces:A Survey of LLM-based Text-to-SQL
  • Flink架构概览,Flink DataStream API 的使用,FlinkCDC的使用
  • 手机充电协议
  • 目标检测135个前沿算法模型汇总(附源码)!
  • rocketmq优先级控制 + 并发度控制
  • 85本适合AI入门的人工智能书籍合集免费资源
  • 游戏引擎学习第301天:使用精灵边界进行排序
  • 数据湖和数据仓库的区别
  • 线程、线程池、异步
  • 人脸识别,使用 deepface + api + flask, 改写 + 调试
  • 【沉浸式求职学习day46】【华为5.7暑期机试题目讲解】
  • 广东省省考备考(第十六天5.21)—言语:语句排序题(听课后强化)
  • Mcu_Bsdiff_Upgrade
  • 数据结构与算法——堆