C语言 之 内存函数 memcpy、memmove函数的使用和模拟实现 memset、memcmp函数的使用

news2025/1/9 23:08:30

文章目录

  • 1.memcpy函数的使用和模拟实现
    • 例子:
    • 模拟实现:
  • 2.memmove函数的使用和模拟实现
    • 例子:
    • 模拟实现:
  • 3.memset函数的使用
    • **例子:**
  • 3.memcmp函数的使用
    • 例子:

首先 我们要明确下面这些函数之所以被称作内存函数,就是因为这些函数所针对的内容是内存空间

1.memcpy函数的使用和模拟实现

函数原型:

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

1.函数memcpy是从source的位置开始向后复制num个字节的数据到destinatio指向的内存位置。(注意此处的num是num个字节,不是num个元素)
2.这个函数在遇到 ‘\0’ 的时候并不会停下来。

例子:

#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是20个字节
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		printf("%d ", arr2[i]);
	}
	return 0;
}

输出结果:
在这里插入图片描述

模拟实现:

#include <stdio.h>
#include <string.h>
#include<assert.h>
void* my_memcpy(void* dst, const void* src, size_t count)
{
	void* ret = dst;  //用来存储初始地址 便于结果的返回
	assert(dst);
	assert(src);
	while (count--) //以count作为条件 即需要拷贝的字节数
	{ 
		*(char*)dst = *(char*)src;  //注意强转为(char*)类型  +1跳过一个字节
		dst = (char*)dst + 1;
		src = (char*)src + 1;
	}
	return ret;
}
int main()
{
	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
	int arr2[10] = { 0 };
	my_memcpy(arr2, arr1, 20);
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		printf("%d ", arr2[i]);
	}
	return 0;
}

当然memcpy的使用会有一个内容重叠的问题,即目标的空间位置和源的空间位置在内存空间中有重叠,造成输出结果的不理想,这里讲的话可能会比较乱,所以晚点有时间再详细写一篇来说明

2.memmove函数的使用和模拟实现

函数原型:

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

1.和memcpy的差别就是memmove函数处理的源内存块和⽬标内存块是可以重叠的。即没有内存重叠的问题。
2.如果源空间和⽬标空间出现重叠,就得使⽤memmove函数处理

例子:

#include <stdio.h>
#include <string.h>
int main()
{
	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
	int arr2[10] = { 0 };
	memmove(arr2, arr1, 20);
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		printf("%d ", arr2[i]);
	}
	return 0;
}

输出结果:
在这里插入图片描述

模拟实现:

为了解决内存重叠的问题,我们可以通过改变拷贝的方式来解决。
memcpy是默认从前往后拷贝的,所以一旦位置出现问题,即dst和src两个指针的位置前后出现问题,就会造成影响
所以为了解决这个问题我们就根据这两个指针的位置的前后不同来进行不同的处理,加上了从后往前拷贝这个方法,就能解决这个问题了。
如果不了解内存重叠,建议先去了解一下。

#include <stdio.h>
#include <string.h>
#include<assert.h>

void* my_memmove(void* dst, const void* src, size_t count)
{
	assert(dst && src);
	void* ret = dst; //用来存储初始地址 便于结果的返回
	if (dst < src)
	{
		//从前往后拷贝
		while (count--)
		{
			*(char*)dst = *(char*)src;
			dst = (char*)dst + 1;
			src = (char*)src + 1;
		}
	}
	else
	{
		//从后往前拷贝
		while (count--)
		{
			*((char*)dst+count) = *((char*)src+count); // 通过+count 来到达后面的位置 
																				//并且count-- 这样就能做到从后往前拷贝
		}
	}

	return ret;
}

int main()
{
	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
	int arr2[10] = { 0 };
	my_memmove(arr2, arr1, 20);
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		printf("%d ", arr2[i]);
	}
	return 0;
}

3.memset函数的使用

函数原型:

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

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

例子:

#include <stdio.h>
#include <string.h>
int main()
{
	char str[] = "hello fall";
	memset(str, 'x', 5);
	printf(str);
	return 0;
}

输出结果:
在这里插入图片描述

3.memcmp函数的使用

函数原型:

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

比较从ptr1和ptr2指针指向的位置开始,向后的num个字节

返回值:
在这里插入图片描述
这个函数和strcmp类似,比较的是ASCII码值,而相等则返回0,ptr1比ptr2大则返回>0的值

例子:

#include <stdio.h>
#include <string.h>
int main()
{
	char buffer1[] = "abd";
	char buffer2[] = "abc";
	int tmp = memcmp(buffer1, buffer2, sizeof(buffer1));
	if (tmp > 0)
		printf("'%s' is greater than '%s'.\n", buffer1, buffer2);
	else if (tmp < 0)
		printf("'%s' is less than '%s'.\n", buffer1, buffer2);
	else
		printf("'%s' is the same as '%s'.\n", buffer1, buffer2);
	return 0;
}

输出结果:
在这里插入图片描述

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

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

相关文章

130-横向移动PTH哈希PTT票据PTK密匙Kerberos

pass the hash &#xff08;哈希传递攻击&#xff0c;简称pth&#xff09;利用lm&#xff08;老版本哈希&#xff09;或ntlm&#xff08;新版本&#xff09;的值进行的渗透测试 pass the ticket &#xff08;票据传递攻击&#xff0c;简称ptt&#xff09;利用的票据凭证TGT进行…

sql手工注入——sqli-labspage第六关到第十关

第六关 一确定攻击点判断闭合方式 跟个双引号引起报错 说明页面有报错语句的位置 http://127.0.0.1/Less-6/?id1%22http://127.0.0.1/Less-6/?id1%22尝试闭合 闭合成功 http://127.0.0.1/Less-6/?id1%22%20--http://127.0.0.1/Less-6/?id1%22%20-- 二.查询数据库名 ht…

多商户小程序审核存在商户入口无法通过

小程序拒绝如下&#xff1a; 需要注意的地方如下&#xff1a; 关闭店铺展示关闭商户入驻关闭diy中的申请入口、店铺街入口等关闭个人中心广告的申请入口关闭分销关闭支付宝

基于Java语言的能源管理系统中软网关的应用

介绍 软网关适用于建筑、工厂、商场、医院、园区、高耗能企业、城市双碳建设平台等的水、电、气、热、油等能源数据采集、加工、分析、预警、碳指标、碳排放计算等场景&#xff1b; 企业通过软网关采集底层电表数据&#xff0c;传输给时序库&#xff0c;从而完成能源管理系统…

nginx全方位讲解

安装nginx [rootnginx ~]# tar zxf nginx-1.24.0.tar.gz [rootnginx ~]# cd nginx-1.24.0/ [rootnginx nginx-1.24.0]# ls auto CHANGES.ru configure html Makefile objs src CHANGES conf contrib LICENSE man README [rootnginx nginx-…

Python编码系列—Python WebSocket 实时通信:构建高效互动的网络应用

&#x1f31f;&#x1f31f; 欢迎来到我的技术小筑&#xff0c;一个专为技术探索者打造的交流空间。在这里&#xff0c;我们不仅分享代码的智慧&#xff0c;还探讨技术的深度与广度。无论您是资深开发者还是技术新手&#xff0c;这里都有一片属于您的天空。让我们在知识的海洋中…

基于贝叶斯优化卷积神经网络(Bayes-CNN)的多因子数据分类识别算法matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 (完整程序运行后无水印) 贝叶斯优化过程 贝叶斯优化后的CNN训练和识别结果 标准的CNN的识别结果 2.算法运行软件版本 matlab2022a 3.部分核心程…

JavaScript(29)——函数参数

动态参数 arguments是函数内部内置的伪数组变量&#xff0c;它包含了调用函数时传入的所有实参&#xff0c;结果是一个伪数组。 function sum() {console.log(arguments);}sum(2, 3, 4)sum(6, 7) 剩余参数 剩余参数允许我们将一个不定数量的参数表示为一个数组 function sum…

深入理解HTTP的基础知识:请求-响应过程解析

首先&#xff0c;我们从网络协议的最顶层开始讲解&#xff0c;即应用层。在网络通信中&#xff0c;应用层是最接近用户的一层&#xff0c;它负责为特定的网络应用提供服务和功能。应用层协议定义了数据交换的规则和格式&#xff0c;以便不同的应用程序能够相互通信和交换信息。…

Xenu 网站死链接检测工具下载以及使用指南

引言 Xenus Link Sleuth&#xff08;简称 Xenu&#xff09;是一款功能强大的网站死链接检测工具。由于其绿色蜗牛形状的图标&#xff0c;Xenu 被广大的 SEO 人员亲切地称为“绿蜗牛”。 随着网站的发展&#xff0c;可能会因为删除过期内容或改版而导致一些死链接的产生。死链…

扫描包得到所有Bean的Class对象

文章目录 1.任务介绍以及类加载器1.介绍2.类加载器1.介绍2.自己程序的类路径 2.新建一个模块sun-spring用来写自己的spring容器1.新建模块2.查看是否交给父模块管理3.引入日志 3.具体实现1.目录2.Component.java3.ComponentScan.java4.MonsterDao.java5.MonsterService.java6.S…

第三期书生大模型实战营 进阶岛第3关LMDeploy 量化部署进阶实践

环境准备 conda create -n lmdeploy python3.10 -y conda activate lmdeploy conda install pytorch2.1.2 torchvision0.16.2 torchaudio2.1.2 pytorch-cuda12.1 -c pytorch -c nvidia -y pip install timm1.0.8 openai1.40.3 lmdeploy[all]0.5.3为方便文件管理&#xff0c;我…

Vue(一) 插值与指令语法、数据代理、MVVM模式、事件处理

文章目录 1. 初始Vue2. 模板语法2.1 插值语法2.2 指令语法 3. el与data的两种写法3.1 el的两种写法3.2 data的两种写法 4. MVVM模式5. 数据代理5.1 Object.defineProperty()5.2 何为数据代理5.4 vue中的数据代理 6. 事件处理6.1 v-on6.2 事件修饰符6.3 键盘按键事件 1. 初始Vue…

【C语言】预处理的使用

预处理 一、预处理-宏定义1、程序编译过程(1) 编写源程序(2) 程序编译过程说明 2、预处理3、宏的概念4、无参宏5、带参宏6、带参宏的副作用7、宏定义中的符号粘贴 二、预处理.条件编译1、无值宏定义2、条件编译3、条件编译的使用场景 三、预处理.头文件1、头文件的作用2. 头文件…

RTSP/Onvif安防视频监控平台EasyNVR在欧拉系统中启动失败的原因排查

视频安防监控平台EasyNVR可支持设备通过RTSP/Onvif协议接入&#xff0c;并能对接入的视频流进行处理与多端分发&#xff0c;包括RTSP、RTMP、HTTP-FLV、WS-FLV、HLS、WebRTC等多种视频流格式。平台支持轻量化部署&#xff0c;可兼容各类操作系统&#xff0c;包括Windows、Linux…

Admin.NET源码学习(4:基于Furion的后台服务启动方式浅析)

Admin.NET为前后端分离架构&#xff0c;后台服务的入口项目为Admin.NET.Web.Entry&#xff0c;其与其它项目的依赖关系如下图所示。   由于项目采用Furion框架&#xff0c;后台服务启动方式、注册方式、配置方式等方面与常规的asp.net core项目差异明显&#xff0c;初步接触…

计算机的错误计算(七十)

摘要 讨论大数的正割函数 sec(x)的错误计算。 例1. 已知 在 Maple 中计算 在 Maple中输入&#xff1a; restart; sec(30^54.8); 则输出&#xff1a; -5.214386310 若输入&#xff1a; Digits : 16;evalf[16](sec(30^54.8)); 则输出&#xff1a; 1.324455078865824…

中年人开发语言学习之路,反其道而行之

大家都更愿意学习新技术、新架构&#xff0c;代表着新方向新趋势&#xff0c;当大家都这么想的时候&#xff0c;注定了竞争就会激烈。有一部分中年程序员&#xff0c;反其道而行之&#xff0c;学习一些老掉牙的开发语言&#xff0c;向哪些近乎被遗忘的老旧系统进军。 市面上依…

一文了解Ansible原理以及常见使用模块

ansible使用手册 1. 简述 Ansible 是一种开源的自动化工具&#xff0c;主要用于配置管理、应用程序部署和任务自动化。 它使用简单的 YAML 语言来定义自动化的任务【playbook】&#xff0c;使得配置和部署变得更加直观和易于管理。 基于SSH协议连接到远程主机来执行指令。 2…

图像数据处理21

五、边缘检测 5.2基于二阶导数的边缘检测 一阶导数&#xff08;如Sobel、Prewitt算子&#xff09;能够捕捉到灰度值的快速变化&#xff0c;但有时会因检测到过多的边缘点而导致边缘线过粗。为了更加精确地定位边缘位置&#xff0c;可以利用二阶导数的零交叉点。零交叉点是是函…