C语言之内存函数篇(3)

news2025/1/11 19:54:34

目录

memcpy

memcpy的使用

memcpy的模拟实现

NO1.

NO2. 

memcpy可否实现重叠空间的拷贝

my_memcpy 

memcpy 

memmove 

memmove

memmove

分析

代码

memset

memset的使用

memcmp

memcmp的使用 

<0

=0

>0 


今天我们继续介绍几个重要的内存操作函数。🙂🙂🙂

在前面一章我们学过strcpy,strcmp等字符串函数,那这里我们来学习与它们功能类似的内存函数。

还是从参数,返回值,头文件,模拟实现等方面来讲解 

memcpy

memcpy - C++ Reference 

 memcpy与strcpy是类似的,但是strcpy仅仅用在字符串拷贝上,内存中的数据,不仅仅是字   符,还有整型,浮点型等等,这里就要用到我们的memcpy

void * memcpy ( void * destination, const void * source, size_t num );
  • memcpy是库函数,从source的位置开始向后拷贝num个字节的数据到desitination的内存位置
  •  C语言标准值规定,memcpy来实现不重叠的内存的拷贝。memcpy达到60分可以实现功能
  •  在我们VS2022这个环境中,memcpy也可以实现重叠内存的拷贝。此时memcpy达到100分
  • 头文件#include<string.h>
  • 参数size_t num是拷贝的字节个数
  • 参数const void * source 是拷贝的字节的起始位置(const防止source修改,void*类型)
  • 参数void * destination 是拷贝到的空间的起始位置(void*类型)
  • 返回类型void *
  • void*类型表示可以接收任何类型的数据,但是不能 解引用 哦
  • 这个函数在遇到 '\0' 的时候并不会停下来。
  • 如果source和destination有任何的重叠,复制的结果都是未定义的 
  • memcpy的模拟实现

memcpy的使用

#include<stdio.h>
int main()
{
	int a[] = { 0 };
	int b[] = { 6,7,8,9,10 };//5✖4
	void* p = memcpy(a, b, 20);
	int i = 0;
	for (i = 0; i < 5 ;i++)
	{
		printf("%d ", a[i]);
	}
	return 0;
}

 

  • 这下无论是什么类型的数据都可以从一个空间拷贝到另外一个空间里去。 

memcpy的模拟实现

#include<stdio.h>
#include<assert.h>
void* my_memcpy(void* destination, const void* source, size_t num)//记住num是字节的个数
{
    assert(dest && src);
	void* ret = destination;
	while (num--)
	{
		*((char*)destination) = *((char*)source);
		(char*)destination = (char*)destination + 1;
		(char*)source = (char*)source + 1;
	}
	return ret;
}
int main()
{
	int a[20] = { 0 };
	int b[] = { 1,2,3,4,5 };//5✖4
	void* p = my_memcpy(a, b, 20);
	int i = 0;
	for (i = 0; i < 5 ;i++)
	{
		printf("%d ", a[i]);
	}
	return 0;
}

 

NO1.

有同学说可以直接一个整型一个整型的拷贝过去,但是问题在于编写库函数的程序员并不会知道我们需要拷贝怎样的类型数据,所以这里我们又可以用到回调函数当时使用的一个方法 

回调函数------->

C语言之指针进阶篇(3)_唐棣棣的博客-CSDN博客 

 

 

NO2. 

*((char*)destination) = *((char*)source);
(char*)destination++;
(char*)source++;

 那有同学这样写可以吗?

 答:当然是不可以的啦。其实关于前置和后置++我们都尽量少去使用。  因为副作用很多。我们也不清楚它的能使用的情况是怎样一回事,所以尽量少用。这里不能使用的原因是:destinationsource强制转化后,再去++,强制转化已经失效了,++时destinationsource已经变成void*类型的啦

++(char*)destination;
++(char*)source;

 虽然以上写法可能可以,但是会存在无法编译的问题。

还有同学问就不能直接++? 

 答:当然不行。回调函数我们那篇博文我们已经讲解过 void*类型的指针变量不能解引用和            计算++ --等等。

memcpy可否实现重叠空间的拷贝

当我们学习了上面memcpy的知识了,有人就动了动自己的小脑袋。🙂🙂

memcpy真的不能实现重叠内存的拷贝吗?

my_memcpy 

#include<stdio.h>
void* my_memcpy(void* destination, const void* source, size_t num)//记住num是字节的个数
{
	void* ret = destination;
	while (num--)
	{
		*((char*)destination) = *((char*)source);
		destination = (char*)destination + 1;
		source = (char*)source + 1;
	}
	return ret;
}
int main()
{
	int b[] = { 1,2,3,4,5 };
	//把1 2 3拷贝到3 4 5
	void* p = my_memcpy(b+2, b, 12);
	int i = 0;
	for (i = 0; i < 5 ;i++)
	{
		printf("%d ", b[i]);
	}
	return 0;
}

那问题到底出现在那里呢?

 

memcpy 

int main()
{
	int b[] = { 1,2,3,4,5 };
	//把1 2 3拷贝到3 4 5
	void* p = memcpy(b+2, b, 12);
	int i = 0;
	for (i = 0; i < 5 ;i++)
	{
		printf("%d ", b[i]);
	}
	return 0;
}

 

综上所述:事实证明只是我们my_memcpy不可以。

 C语言标准值规定,memcpy来实现不重叠的内存的拷贝。memcpy达到60分可以实现功能

 在我们VS2022这个环境中,memcpy也可以实现重叠内存的拷贝。此时memcpy达到100分

 C语言标准值规定,memmove来实现重叠的内存拷贝。 

memmove 

使用memmove是否可以实现?

//当然可以
int main()
{
	int b[] = { 1,2,3,4,5 };
	//把1 2 3拷贝到3 4 5
	void* p = memmove(b+2, b, 12);
	int i = 0;
	for (i = 0; i < 5 ;i++)
	{
		printf("%d ", b[i]);
	}
	return 0;
}

 

memmove

memmove - C++ Reference (cplusplus.com)

void * memmove ( void * destination, const void * source, size_t num );
  •  C语言标准值规定,memcpy来实现不重叠的内存的拷贝。memcpy达到60分可以实现功能
  •  在我们VS2022这个环境中,memcpy也可以实现重叠内存的拷贝。此时memcpy达到100分
  •  C语言标准值规定,memmove来实现重叠的内存拷贝。
  • memmove也是是库函数,类似memcpy,基本一致所不在详细介绍。
  • 和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。
  • 如果源空间和目标空间出现重叠,就得使用memmove函数处理。
  • memmove的模拟实现

memmove

分析

代码

memset

memset - C++ Reference (cplusplus.com) 

void * memset ( void * ptr, int value, size_t num );
  • memset是库函数,是设置内存的,是以字节为单位设置内存的
  • 头文件#include<string.h>
  • 参数size_t num是设置的字节个数
  • 参数int value 是设置的字节
  • 参数void * ptr是设置的空间的起始位置(void*类型)
  • 返回类型void *
  •  void*类型表示可以接收任何类型的数据,但是不能 解引用 哦和计算++,--等
  • 准确来说memset更加适合char类型的变量,但是可以把整型设置全设为1
  • 注意char可以放到int里,int不能放到char里

memset的使用

#include<stdio.h>
#include<string.h>
int main()
{
	char arr[] = "tsqxgd.haha";
	memset(arr+6, 'x', 3);//再次强调是3个字节
	printf("%s\n", arr);
}

 

memcmp

memcmp - C++ Reference (cplusplus.com)

int memcmp ( const void * ptr1,const void * ptr2,size_t num );
  • memcmp是库函数,是比较内存的,是以字节为单位比较内存的
  • 比较从ptr1和ptr2指针开始的num个字节
  • 头文件#include<string.h>
  • 参数size_t num是设置的字节个数
  • 参数const void * ptr1 是比较的字节的起始位置(const防止source修改,void*类型)
  • 参数const void * ptr2 是比较的空间的起始位置(void*类型)
  •  void*类型表示可以接收任何类型的数据,但是不能 解引用 哦和计算++,--等
  • 返回类型int 
  •  返回值如下图:

memcmp的使用 

<0

#include<stdio.h>
#include<string.h>
int main()
{
	int arr1[] = { 1,2,3,4,5 };
	int arr2[] = { 1,2,3,4,6 };
	int ret = memcmp(arr1, arr2, 20);
	printf("%d\n", ret);
}

 

=0

#include<stdio.h>
#include<string.h>
int main()
{
	int arr1[] = { 1,2,3,4,5 };
	int arr2[] = { 1,2,3,4,6 };
	int ret = memcmp(arr1, arr2, 16);
	printf("%s\n", ret);
}

 

>0 

#include<stdio.h>
#include<string.h>
int main()
{
	int arr1[] = { 1,2,3,4,7 };
	int arr2[] = { 1,2,3,4,6 };
	int ret = memcmp(arr1, arr2,20 );
	printf("%d\n", ret);
}

 

特别提醒:

只是在vs2022编译器下,才是0,1,-1这样的数值,

在其他编译器下可能就不是但是肯定是<0 >0 =0这样的范围的。 

✔✔✔✔✔最后,感谢大家的阅读,若有错误和不足,欢迎指正!

我知青山不复在,也知绿水难长流。

下篇博文我们总结各种函数的模拟实现。🆗🆗🆗🆗走的每一步都是算数的。

代码------→【gitee:唐棣棣 (TSQXG) - Gitee.com】

联系------→【邮箱:2784139418@qq.com】

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

js中的数据结构:栈,队列,链表,字典哈希表,树

栈&#xff1a;先进后出 队列&#xff1a;先进先出 链表&#xff1a; 单链表&#xff1a; 双链表&#xff1a; 环形链表&#xff1a;最后一个数据的next指针不是指向null&#xff0c;指向的是任意之间的一个数据&#xff0c;形成一个环 数组和链表的区别&#xff1a; 字典和哈…

FPGA的汽车尾灯控制Verilog

名称&#xff1a;汽车尾灯控制Verilog 软件&#xff1a;Quartus 语言&#xff1a;Verilog 要求&#xff1a; 设计一个汽车尾灯控制器。假设汽车尾部各有2个指示灯&#xff08;LED&#xff09;&#xff0c;分别代表转弯、刹车&#xff0c;控制器功能包括&#xff1a; &…

微信收款码费率0.38太坑了

作为一个有多年运营经验的商家&#xff0c;我本人在申请收款功能时曾经走过了不少弯路。我找遍了市面上的知名的支付公司&#xff0c;但了解到的收款手续费率通常都在0.6左右&#xff0c;最低也只能降到0.38。这个过程吃过不少苦头。毕竟&#xff0c;收款功能是我们商家的命脉&…

有什么好用的设备管理软件?智慧巡检对后勤运维有什么帮助?

定期巡检在设备管理、后勤管理和运维管理中扮演着不可或缺的角色&#xff0c;以及及时发现异常并控制风险。然而&#xff0c;传统巡检工作存在弊端。为了解决这些问题&#xff0c;“的修”报修系统对巡检管理功能进行了突破性的设计和开发。   “的修”报修巡检管理功能包含了…

C++之内部类实现总结(二百三十五)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

vscode终端中打不开conda虚拟包管理

今天&#xff0c;想着将之前鸽的Unet网络模型给实现一下&#xff0c;结果发现&#xff0c;在vscode中运行python脚本&#xff0c;显示没有这包&#xff0c;没有那包。但是在其他的ipynb中是有的&#xff0c;感觉很奇怪。我检查了一下python版本&#xff0c;发现不是我深度学习的…

转载 - 洞察问题本质,解决工作难题

作者&#xff1a;关苏哲 高效管理者的三大技能 问题界定的6个问题 1.你所需要解决的问题是什么&#xff1f; 2.你为什么需要解决这个问题&#xff1f; 3.你期待的理想结果是什么&#xff1f; 4.这个问题包括哪些子问题&#xff1f; 5.你曾经尝试过哪些解决方式&#xff1f…

Intel架构的基本知识

字节序 字节序根据存储的方向不同, 分为大端字节序(big-endian) 和 小端字节序(little-endian) 大端字节序(big-endian): 低字节存储在起始地址 小端字节序(little-endian): 高字节存储在起始地址 获得当前主机字节序 #include <iostream>using namespace std;int ma…

LabVIEW崩溃后所产生的错误日志文件的位置

LabVIEW崩溃后所产生的错误日志文件的位置 LabVIEW开发环境刚刚崩溃&#xff0c;请问我如何访问崩溃后自动生成的日志文件&#xff1f; LabVIEW崩溃后产生的转储文件位于何处&#xff1f; 代码导致了LabVIEW崩溃&#xff0c;请问哪些文件可以帮助NI技术支持了解具体原因&…

如何做到人声和背景音乐分离?简单粗暴,赶紧学起来~

在这个短视频盛行的时代&#xff0c;优质的背景音乐会让视频锦上添花&#xff0c;但也会造成类似的问题&#xff1a;想单独使用视频中的某一段人声&#xff0c;但会被背景音乐扰乱视听效果。这时就需要将人声和背景音乐进行分离了&#xff0c;今天就来教大家如何将人声和背景音…

最新文档:微信、企业号+地理位置定位+地图展示

概述 公司有需求&#xff0c;通过企业号打卡项目&#xff0c;需要用到企业微信定位 详细 前言 demo是基于微信、企业号平台的一个定位&#xff0c;地图展示项目 后台使用springboot架构搭建的与微信交互的服务&#xff0c;使用httpclient连接池&#xff0c;调用微信接口&…

小黑子—MyBatis:第二章

MyBatis入门2.0 四 小黑子诉说Mybatis核心配置文件详情4.1 多环境4.2 Mybatis的事务管理器 - transactionManager4.3 dataSource&#xff08;数据源&#xff09;4.3.1 不同类型下的数据源有不同的属性4.3.2 pool 和 unpooled 的区别4.3.3 配置具体的数据库连接池对象 4.4 prope…

【操作系统笔记三】内存寻址

物理寻址 主存&#xff08;内存&#xff09; 计算机主存也可以称为物理内存&#xff0c;内存可以看成由若干个连续字节大小的单元组成的数组每个字节都有一个唯一的物理地址&#xff08;Physical Address&#xff09;CPU访问内存前&#xff0c;先拿到内存地址&#xff0c;然后…

Django — 会话

目录 一、Cookie1、介绍2、作用3、工作原理4、结构5、用途6、设置7、获取 二、Session1、介绍2、作用3、工作原理3、类型4、用途5、设置6、获取7、清空信息 三、Cookie 和 Session 的区别1、存储位置2、安全性3、数据大小4、跨页面共享5、生命周期6、实现机制7、适用场景 四、P…

掌动智能浅析故障注入测试的好处与实践方法

在现代技术环境中&#xff0c;系统面临各种潜在的威胁和故障&#xff0c;如硬件故障、网络问题、软件错误等。为了应对这些挑战&#xff0c;开发团队需要确保系统在逆境中依然能够提供可靠的服务。故障注入测试是一种模拟现实故障和异常情况的方法&#xff0c;旨在提高系统的鲁…

vue3+ts+java使用WebSocket传输数据

一、环境 系统&#xff1a;win11 IDE&#xff1a;vscode 框架&#xff1a;electron22.0.0vite2vue3typescript4.8.4springboot2.2.5jdk1.8 二、websocket介绍 2.1 由来 WebSocket未出现之前&#xff0c;浏览器和服务器之间的通信是通过Web的poll技术进行通信&#xff0c;就…

牛客java训练题 day1

9.24 day1 Q 1. this 指针是用来干什么的&#xff1f; 2.基类和派生类分别是指什么&#xff1f; 3.为什么方法中不能写静态变量 4. 解释一下ASCII码和ANSI码和两者的区别 5.简述j ava.io java.sql java.awt java.rmi 分别是什么类型的包 6. 看下面一段代码&#xff1a;…

分类预测 | Matlab实现NGO-CNN-SVM北方苍鹰算法优化卷积支持向量机分类预测

分类预测 | Matlab实现NGO-CNN-SVM北方苍鹰算法优化卷积支持向量机分类预测 目录 分类预测 | Matlab实现NGO-CNN-SVM北方苍鹰算法优化卷积支持向量机分类预测分类效果基本描述程序设计参考资料 分类效果 基本描述 1.Matlab实现NGO-CNN-SVM北方苍鹰算法优化卷积支持向量机分类预…

安全防御第二次作业

1. 防火墙支持那些NAT技术&#xff0c;主要应用场景是什么&#xff1f; 防火墙支持几乎所有的NAT技术&#xff0c;包括源NAT、目标NAT、双向NAT等&#xff0c;主要应用场景是保护内部网络免受外部网络的攻击 NAT技术可以将IP数据报文头中的IP地址转换为另一个IP地址&#xff…