C语言第二十七弹---内存函数

news/发布时间2024/5/14 5:48:41

个人主页: 熬夜学编程的小林

💗系列专栏: 【C语言详解】 【数据结构详解】

内存函数

1、memcpy 使用和模拟实现

2、memmove 使用和模拟实现

3、memset 函数的使用

4、memcmp 函数的使用

总结


前面两弹讲解了字符函数和字符串函数,但是在我们实际运用中不仅仅只有这些函数,因此下面我们继续需要几个常见的内存函数。

1、memcpy 使用和模拟实现

void * memcpy ( void * destination, const void * source, size_t num );
函数memcpy从source的位置开始向后复制num个字节的数据到destination指向的内存位置。
num为拷贝的字节数。
这个函数在遇到 '\0' 的时候并不会停下来。
如果source和destination有任何的重叠,复制的结果都是未定义的。
返回值为目标空间首地址。
#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);int i = 0;for (i = 0; i < 10; i++){printf("%d ", arr2[i]);}return 0;
}
对于重叠的内存,交给memmove来处理。
但是在VS 2022中memcpy也可以解决重叠内存情况。
memcpy函数的模拟实现:
根据前面两弹讲的五个函数模拟实现考虑点,依次进行思考

1.参数顺序

2.const修饰指针(防止指针被修改)
3.函数的功能,停止条件

4.assert(对空指针进行判断)
5.函数返回值

思想:从前往后依次按照char类型拷贝,不能解决内存重叠问题。
void * 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;//强转成char*类型,再解引用给目标空间dst = (char *)dst + 1;//目标空间向前移动src = (char *)src + 1;//原来空间向前移动}return ret;
}

2、memmove 使用和模拟实现

void * memmove ( void * destination, const void * source, size_t num );
和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。
如果源空间和目标空间出现重叠,就得使用memmove函数处理。
#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);int i = 0;for (i = 0; i < 10; i++){printf("%d ", arr2[i]);}return 0;
}
输出的结果:
memmove的模拟实现:
要处理内存重叠问题,因此需要考虑两种情况,如果内存不重叠,从前往后拷贝即可,如果内存重叠,可以从后往前拷贝。
void * memmove ( void * dst, const void * src, size_t count)
{void * ret = dst;if (dst <= src || (char *)dst >= ((char *)src + count)) {/** 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);
}

3、memset 函数的使用

void * memset ( void * ptr, int value, size_t num );
memset是用来设置内存的,将内存中的值以字节为单位设置成想要的内容。
ptr 指向要填充的内存块的指针。

value为要设置的值。该值以 int 形式传递,但该函数使用此值的无符号 char 转换填充内存块。

num 为要设置为值的字节数。size_t 是一个
无符号的整数类型。
#include <stdio.h>
#include <string.h>
int main ()
{char str[] = "hello world";memset (str,'x',6);printf(str);return 0;
}
输出的结果:

4、memcmp 函数的使用

int memcmp ( const void * ptr1, const void * ptr2, size_t num );
较从ptr1和ptr2指针指向的位置开始,向后的num个字节

比较两个内存块
将 ptr1 指向的内存块的第一个 num 个字节与 ptr2 指向的第一个 num 个字节进行比较,如果它们都匹配,则返回零,如果它们不匹配,则返回一个与零不同的值,表示哪个值更大。

请注意,与 strcmp 不同,该函数在找到 null 字符后不会停止比较。

返回值如下:
<0  在两个内存块中不匹配的第一个字节在 ptr1 中的值低于 ptr2 中的值(如果计算为 unsigned char 值)
0    两个存储块的内容相等
>0  两个内存块中不匹配的第一个字节在 ptr1 中的值大于 ptr2 中的值(如果计算为无符号字符值)
#include <stdio.h>
#include <string.h>
int main()
{char buffer1[] = "DWgaOtP12df0";char buffer2[] = "DWGAOTP12DF0";int n;n = memcmp(buffer1, buffer2, sizeof(buffer1));if (n > 0) printf("'%s' is greater than '%s'.\n", buffer1, buffer2);else if (n < 0) printf("'%s' is less than '%s'.\n", buffer1, buffer2);elseprintf("'%s' is the same as '%s'.\n", buffer1, buffer2);return 0;
}

总结


本篇博客就结束啦,谢谢大家的观看,如果公主少年们有好的建议可以留言喔,谢谢大家啦!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.bcls.cn/pMYr/2811.shtml

如若内容造成侵权/违法违规/事实不符,请联系编程老四网进行投诉反馈email:xxxxxxxx@qq.com,一经查实,立即删除!

相关文章

http前生今世

HTTP/0.9&#xff0c;仅支持GET方法&#xff0c;并且响应中没有HTTP头信息&#xff0c;只有文档内容。 HTTP/1.0增加了对POST方法、状态码、HTTP头信息等的支持&#xff0c;这一版本也是广泛应用的历史性版本。 HTTP/1.1引入了持久连接&#xff08;Persistent Connections&…

时间函数,SQL获取,实际案例,简单易懂

时间函数&#xff1a; TODAY(): 返回当前日期 DATE():返回表示特定日期的连续序列号 Time&#xff1a;返回指定时间的序列号。 Time("时","分","秒") EOMONTH():以日期/时间格式返回指定月份数之前或之后的月份的最后一天的日期 EOMONTH(…

SparkSQL学习02-编程入口

文章目录 1 DataFrame的构建方式方式一&#xff1a;JavaBean反射的方式1.1 创建Scala类1.2 创建Scala对象 方式二&#xff1a;动态编码的方式 2 DataSet的构建方式3 RDD和DataFrame以及DataSet之间的相互转换3.1【RDD-->DataFrame】和【RDD-->DataSet】3.2【DataFrame--&…

Matlab/simulink光伏发电的恒定电压法MPPT仿真(持续更新)

1.光伏发电的电导增量法MPPT仿真 2.光伏发电的恒定电压法MPPT仿真 3.光伏发电的扰动观察法MPPT仿真 4.光伏发电的占空比法MPPT仿真 5.基于神经网络的MPPT光伏发电仿真 6. 基于模糊控制的MPPT光伏发电仿真 7. 基于粒子群算法&#xff08;PSO&#xff09;的500w光伏系统MPPT控…

【2024.02.23】定时执行专家V7.0最新版GUI界面 - 基于wxWidgets 3.2.4,开发工具CodeBlocks + GCC9.2.0

目录 ◆ V7.0 更新日志 ◆ 主窗口 ◆ 关于对话框 ◆ 任务对话框 ◆ 触发器对话框 ◆ 设置对话框 ◆ 选择语言对话框 ◆ V7.0 更新日志 定时执行专家V7.0 龙年春节版 ★★★★★★★★★★★★★★★★★★★★ ▼2024-02-22 V7.0 - 更新日志▼ - 增加 执行记录功能&am…

一文带你了解如何在Java中操作Redis

文章目录 前言发现宝藏一、 Redis客户端简介1. Redis客户端分类2. Spring 整合 Redis 的两种方式 二、 使用 Jedis 操作 Redis1. Jedis的maven坐标2. 使用Jedis操作Redis的步骤3. Jedis 操作 Redis 示例 三、 使用 Spring Data Redis 操作 Redis1. Spring Data Redis 的 maven …

区块链游戏解说:什么是 Planet IX

作者&#xff1a;lesleyfootprint.network 编译&#xff1a;cicifootprint.network 数据源&#xff1a;Planet IX Dashboard 什么是 Planet IX Planet IX&#xff0c;一个由原生 IX TOKEN 推动的 Web3 玩赚平台。作为一款 GameFi 策略游戏&#xff0c; Planet IX 上的每项资…

四、分类算法 - sklearn转换器和估算器

目录 1、sklearn转换器和估算器 1.1 转换器 - 特征工程的父类 1.2 估计器 - sklearn机器学习算法的实现 目标值&#xff1a;分类 sklearn转换器和估算器KNN算法模型选择和调优朴素贝叶斯算法决策树随机森林 1、sklearn转换器和估算器 1.1 转换器 - 特征工程的父类 1.2 估…

用纯HTML写一个凭证并打印

最近有个需求&#xff0c;需要通过网页把单子打印出来&#xff0c;就用html实现了一个&#xff0c;主要使用了windwos自带的print打印&#xff0c;全部代码如下&#xff1a; <!DOCTYPE html> <html><head><meta http-equiv"Content-Type" cont…

代码随想录算法刷题训练营day22

代码随想录算法刷题训练营day22&#xff1a;LeetCode(236)二叉树的最近公共祖先、LeetCode(235) 二叉搜索树的最近公共祖先、LeetCode(701)二叉搜索树中的插入操作、LeetCode(450)删除二叉搜索树中的节点 LeetCode(236)二叉树的最近公共祖先 题目 代码 /*** Definition for…

Vue的个人笔记

Vue学习小tips ctrl s ----> 运行 alt b <scrip> 链接 <script src"https://cdn.jsdelivr.net/npm/vue2.7.16/dist/vue.js"></script> 插值表达式 指令

C#_扩展方法

简述&#xff1a; 扩展方法所属类必需是静态类&#xff08;类名依据规范通常为XXXExtension&#xff0c;XXX为被扩展类&#xff09;扩展方法必需是公有的静态方法扩展方法的首个参数由this修饰&#xff0c;参数类型为被扩展类型 示例&#xff1a; static class DoubleExtens…

Springboot打包、部署

一、导入maven打包插件 <build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins> </build> 二、执行打包操作&#xff08;…

LiteOS-M代码分析-系统初始化los_init.c

目录 一、LiteOS-M 初始化内核二、LOS_KernelInit代码分析三、LOS_Start代码解析坚持就有收获 一、LiteOS-M 初始化内核 在LiteOS-M应用程序中&#xff0c;系统初始化如下&#xff1a; /*** brief This is the ohos entry, and you could call this in your main funciton af…

【教3妹学编程-算法题】匹配模式数组的子数组数目 I

3妹&#xff1a;2哥2哥&#xff0c;你有没有看到上海女老师出轨男学生的瓜啊。 2哥 : 看到 了&#xff0c;真的是太毁三观了&#xff01; 3妹&#xff1a;是啊&#xff0c; 老师本是教书育人的职业&#xff0c;明确规定不能和学生谈恋爱啊&#xff0c;更何况是出轨。 2哥 : 是啊…

【Pytorch深度学习开发实践学习】B站刘二大人课程笔记整理lecture05 构建线性模型

lecture05 构建线性模型 课程网址 Pytorch深度学习实践 部分课件内容 import torchx_data torch.tensor([[1.0],[2.0],[3.0]]) y_data torch.tensor([[2.0],[4.0],[6.0]])class LinearModel(torch.nn.Module):def __init__(self):super(LinearModel, self).__init__()self.lin…

LeetCode每日刷题:101. 对称二叉树

题目&#xff1a; 解题思路&#xff1a;可以新写一个函数&#xff0c;从root开始&#xff0c;root的left的头结点将记为lefttree&#xff08;左子树&#xff09;,root的lright的头结点将记为righttree&#xff08;右子树&#xff09;&#xff0c; 然后递归左子树的root.left与右…

智能合约,数据资产变现金的一把金钥匙?

大数据产业创新服务媒体 ——聚焦数据 改变商业 在数字时代的黎明&#xff0c;数据不仅是知识的载体&#xff0c;更成为了经济增长的新引擎。在这个由数据驱动的世界里&#xff0c;数据资产如同新发现的矿藏&#xff0c;蕴藏着无限的潜力和价值。 然而&#xff0c;随着这些数字…

【机器学习笔记】 15 机器学习项目流程

机器学习的一般步骤 数据清洗 数据清洗是指发现并纠正数据文件中可识别的错误的最后一道程序&#xff0c;包括检查数据一致性&#xff0c;处理无效值和缺失值等。与问卷审核不同&#xff0c;录入后的数据清理一般是由计算机而不是人工完成。 探索性数据分析(EDA 探索性数据…
推荐文章