内存操作函数

在这里插入图片描述

前言

🎈个人主页:🎈 :✨✨✨初阶牛✨✨✨
🐻推荐专栏: 🍔🍟🌯 c语言进阶
🔑个人信条: 🌵知行合一
🍉本篇简介:>:介绍c语言中有关指针更深层的知识.
金句分享:

✨未来如星辰大海般璀璨✨
✨不必踌躇于过去的半亩方塘.✨

目录

  • 前言
  • 一、memcpy函数
    • 函数模型:
    • 函数参数:
    • 函数的应用:
    • 函数的模拟实现:
  • 二、memmove函数
    • 函数模型:
    • 函数参数:
    • 函数模拟实现
  • 三、memcmp函数
    • 函数模型:
    • 函数参数:
    • 函数应用:
    • 模拟实现
  • 四、memset函数
    • 函数模型:
    • 函数参数

一、memcpy函数

函数模型:

在这里插入图片描述

函数参数:

参数意义
destination指向目的地地址的指针,类型转换为 void* 类型的指针。
source指向要复制的数据源的指针,类型转换为 const void* 类型的指针。
num要复制的字节数

函数头文件:

#include <string.h>

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

注意:
要保证目的地址存的下源地址的内容,要守规矩哦!

1.为什么有了strcpy函数还要有memcpy函数呢?

因为strcpy函数是字符串拷贝,包括strncpy函数也是一样,只能拷贝字符串,对于其它类型,比如整形的拷贝就无能为力.

2.为什么要设置为void*类型?

如果看过前面qsort函数的模拟,这里应该也不难理解,因为要拷贝的类型是未知的,为了能够接收所有类型,所以设置为void*,使用时强制转换为char*类型,然后按字节个数拷贝即可.

函数的应用:

1.将arr2数组的前三个元素拷贝到arr1数组的前三个位置.
2.将字符串str2全部除了’\0’以外,拷贝到str1字符串.

#include <stdio.h>
#include <string.h>
int main()
{
	int arr1[10] = { 6,6,6,6,6,6,6,6,6,6 };
	int arr2[] = { 1,2,3,4,5,6,7,8 };
	int arr2sz= sizeof(arr2) / sizeof(arr2[0]);
	char str1[] = "XXXXXXXXXXXXXXXX";
	char str2[] = "Hello CSDN!";
	int str2sz= sizeof(str2) / sizeof(str2[0]);
	memcpy(arr1, arr2, sizeof(int) * 3);
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr1[i]);
	}
	printf("\n");
	memcpy(str1, str2, sizeof(char) * str2sz - 1);//-1是因为这里使用的是sizeof()函数计算的长度,会包括\0
	printf("%s", str1);
	printf("\n");
	return 0;
}

运行结果:

1 2 3 6 6 6 6 6 6 6
Hello CSDN!XXXXX

函数的模拟实现:

#include<assert.h>
void* my_memcpy(void* destination, const void* source, size_t num)
{
	assert(destination);
	assert(source);
	while (num--)
	{
	//void*类型不可直接使用,要强制转化
		*(char*)destination = *(char*)source;//*是为了找到该地址所对应的字符串
		((char*)destination)++;//指针分别往后走,所以不需要解引用
		((char*)source)++;
	}
}

思考:
memcpy函数可以拷贝内存重叠的地址吗?
即source和destination有重叠的时候.(可以试试).

二、memmove函数

函数模型:

在这里插入图片描述

函数参数:

参数意义
destination指向目的地地址的指针,类型转换为 void* 类型的指针。
source指向要复制的数据源的指针,类型转换为 const void* 类型的指针。
num要复制的字节数

函数作用:
该函数与memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠
的。
如果源空间和目标空间出现重叠,就得使用memmove函数处理,其实在有的编译器中,对于memcpy函数也进行了优化,也是允许内存重叠的.

但牛牛认为,还是有必要掌握这种更"稳妥"的函数的.

情况1

在这里插入图片描述
这种情况我们可以正常的从前向后(前->后) 拷贝.
即:先将6拷贝至2位置,再讲7拷贝至3位置……

在这里插入图片描述
这种情况我们使用从从前向后(前->后)拷贝则达不到我们的要求.
如果我们先将6拷贝至8处,7拷贝至9处,则原来的8 9都被修改了.
这时我们需要从后往前(后->前)拷贝.

即先将11拷贝至13处,再将9拷贝至12处……

在这里插入图片描述
其实这种情况并不是内存地址重叠,目的地址出现在前面也是一样的,都是可以从前向后(前->后)拷贝和从后往前(后->前)拷贝皆可.

函数模拟实现

#include <stdio.h>
#include<assert.h>
void* my_memmove(void* destination, const void* source, size_t num)
{
	void* ret = destination;//保留目的地原来的首地址
	assert(destination && source);//防止传错地址
	if (destination < source)//情况1时,目的地址的首地址小于源地址首地址
	{
		//前-->后
		while (num--)
		{
			*(char*)destination = *(char*)source;
			((char*)destination)++;
			((char*)source)++;
		}
	}
	else
	{
		//后->前
		while (num--)
		{
			*((char*)destination + num) = *((char*)source + num);//注意,这里如果以destination为1的偏移量,+num则是目的地地址的最后一个元素的地址.
		}
	}
	return ret;
}

代码测试:
情况1;

int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,11,12,13 };
	//情况1
	my_memmove(arr + 1, arr + 5, sizeof(int) * 5);
	int sz= sizeof(arr) / sizeof(arr[0]);
	for (int i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

运行结果:

1 6 7 8 9 11 7 8 9 11 12 13

情况2:


int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,11,12,13 };
	//情况1
	my_memmove(arr +7, arr + 5, sizeof(int) * 5);
	int sz = sizeof(arr) / sizeof(arr[0]);
	for (int i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

运行结果:

1 2 3 4 5 6 7 6 7 8 9 11

三、memcmp函数

函数模型:

在这里插入图片描述

函数参数:

参数意义
ptr1指向要比较的内容1,指针
ptr2指向要比较的内容2,指针
num要比较的字节个数

函数头文件:

#include <string.h>

函数功能
比较从ptr1和ptr2指针开始的num个字节.

注意与strcmp函数区分:

strcmp函数:用于比较字符串是否相等,只用于比较字符

memcmp:按字节比较,两个指针所指向的内容是否相等.可用于比较不同类型的元素,但是是按字节比较.

返回值意义
<0第一个不想等的字节中的值的str1的值小于str2中的值(arr1<arr2)
0两个指针的内容相等
>0第一个不相等的字节的str1的值大于于str2中的值(arr1>arr2

函数应用:

试着猜一下这段代码的值?(vs2022环境(小端))

//memcmp函数模拟
#include <stdio.h>
#include <string.h>
int main()
{
	int arr1[] = { 1,4,7,255 };
	int arr2[] = { 1,4,7,511 };
	int ret=memcmp(arr1, arr2, 13);
	printf("%d", ret);
	return 0;
}

在这里插入图片描述
答案:

0

解析:
首先这里是小端存储模式.则这两个数组在内存中存储的是:

arr1 :01 00 00 00 04 00 00 00 07 00 00 00 FF 00 00 00 (16进制,两位代表一个字节)
arr2 :01 00 00 00 04 00 00 00 07 00 00 00 FF 10 00 00

很明显,这里前四个元素(1,4,7)是相等的,而元素255和元素511则是第一个字节的内容相等,所以当比较的元素是[0,13]时,memcmp函数会认为这两个指针所指向的内容是相同的,故返回值是0.

可以试着改成14字节,返回值会是-1,因为第14个字节,arr1是00.arr2是1,则arr1<arr2;

模拟实现

#include<assert.h>
int my_memcmp(const void* ptr1, const void* ptr2, size_t num)
{
	assert(ptr1 && ptr2);
	while (num--)//比较num个字节
	{
		if (*(char*)ptr1 == *(char*)ptr2)//相等继续往后比较
		{
			((char*)ptr1)++;
			((char*)ptr2)++;
		}
		else if (*(char*)ptr1 > *(char*)ptr2)//str1大则返回大于0的数
		{
			return 1;
		}
		else//str2大返回小于0的数
		{
			return -1;
		}
	}
	return 0;//比完了num个字节都相等则返回0
}

四、memset函数

函数模型:

在这里插入图片描述
头文件:

#include <string.h>

函数参数

参数意义
ptr指向要设置内容的空间
value要设置的值,虽然是整形类型,但是也可以是字符,字符会转化为对应的ASCII码值
num要设置空间内容的字节个数

函数功能:
将ptr所指向的空间,的num个字节设置为value值,注意这里也是按字节设置.

示例:

#include <stdio.h>
#include <string.h>
int main()
{
	int arr[5] = { 0 };
	memset(arr, 1, 20);
	for (int i = 0; i < 5; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

结果会是下面这个吗?

1 1 1 1 1

正确答案:

16843009 16843009 16843009 16843009 16843009

为什么呢?
我们可以试着通过调试找到答案.

执行语句memset(arr, 1, 40);后

很显然,这里是将20个字节设置为了1,而一个整形占4个字节;

那么怎样才可以将每个元素都设置为1呢?

#include <stdio.h>
#include <string.h>
int main()
{
	int arr[5] = { 0 };
	for (int i = 0; i < 5; i++)
	{
		memset(arr+i, 1, 1);
	}
	for (int i = 0; i < 5; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

其实这样才能达到我们想要的结果,这里是将一个字节设置为1,然后跳过四个字节(下一个元素),再将这个元素的首字节设置为1.
但是这样是不是有点太麻烦了,所以一般memset函数一般不会这么用,一般memset函数用来将一段空间的元素按字节都初始化为0,在后面我们用到malloc函数向内存申请空间时,得到的空间里面存放的值都是随机值,这时我们可以使用它来帮助我们完成初始化操作.

好了,今天的内存操作函数就讲到这里了,我们下周再见!
最后,如果文章对大家有帮助的话,求一波三连吧!
💗💗💗886
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

蓝桥杯Web前端练习-----渐变色背景生成器

介绍 相信做过前端开发的小伙伴们对渐变色在 UI 设计中的流行度一定不陌生&#xff0c;网页上也时常可以看到各类复杂的渐变色生成工具。使用原生的 CSS 变量加一些 JS 函数就能做出一个简单的渐变色背景生成器。 现在渐变色生成器只完成了颜色选取的功能&#xff0c;需要大家…

【你不知道的 CSS】你写的 CSS 太过冗余,以至于我对它下手了

:is() 你是否曾经写过下方这样冗余的CSS选择器: .active a, .active button, .active label {color: steelblue; }其实上面这段代码可以这样写&#xff1a; .active :is(a, button, label) {color: steelblue; }看~是不是简洁了很多&#xff01; 是的&#xff0c;你可以使用…

5种最佳像素化图像的方法

5种最佳像素化图像的方法1. 什么是像素化&#xff1f;2. 像素化有什么用&#xff1f;3. 如何像素化图像&#xff1f;参考Pixelate 像素化 这篇博客将讨论像素化及如何以五种最佳方式对图像进行像素化。有时希望在在线共享照片时保护照片的隐私。因此在共享图像之前会对图像的某…

锂电池充电的同时也能放电吗?

大家应该都有这样经历&#xff0c;我们的手机在充电的同时也能边使用&#xff0c;有的同学就会说了&#xff0c;这是因为手机电池在充电的同时也在放电。如果这样想我们可能就把锂电池类比了一个蓄水池&#xff0c;以为它在进水的同时也能出水&#xff0c;其实这个比喻是错误的…

【深度强化学习】(5) DDPG 模型解析,附Pytorch完整代码

大家好&#xff0c;今天和各位分享一下深度确定性策略梯度算法 (Deterministic Policy Gradient&#xff0c;DDPG)。并基于 OpenAI 的 gym 环境完成一个小游戏。完整代码在我的 GitHub 中获得&#xff1a; https://github.com/LiSir-HIT/Reinforcement-Learning/tree/main/Mod…

【洛谷刷题】蓝桥杯专题突破-深度优先搜索-dfs(10)

目录 写在前面&#xff1a; 题目&#xff1a;P1019 [NOIP2000 提高组] 单词接龙 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 题目描述&#xff1a; 输入格式&#xff1a; 输出格式&#xff1a; 输入样例&#xff1a; 输出样例&#xff1a; 解题思路&#xff1a; 代…

【数据结构】顺序表和链表

目录.顺序表.链表比较.顺序表 线性表的顺序存储结构&#xff0c;使用一段物理地址连续的存储单元以此存储数据单元的线性结构&#xff08;从头开始连续存储&#xff09; 静态顺序表&#xff1a;使用定长数组存储动态顺序表&#xff1a;使用动态开辟的数组存储&#xff08;常用…

第十三届蓝桥杯省赛 python B组复盘

文章目录前言主要内容&#x1f99e;试题 A&#xff1a;排列字母思路代码&#x1f99e;试题 B&#xff1a;寻找整数思路代码&#x1f99e;试题 C&#xff1a;纸张尺寸思路代码&#x1f99e;试题 D&#xff1a;数位排序思路代码&#x1f99e;试题 E&#xff1a;蜂巢思路代码&…

打印菱形、三角形-课后程序(JavaScript前端开发案例教程-黑马程序员编著-第2章-课后作业)

【案例2-10】打印菱形、三角形 一、案例描述 考核知识点 for双重循环 练习目标 掌握for循环应用。打印出菱形打印出三角形。 需求分析 在本案例中我们将用JavaScript代码在页面中用“*”打印出菱形和三角形。 案例分析 菱形效果如图2-16所示。输入菱形行数6打印菱形 三角形…

计及光伏波动性的主动配电网有功无功协调优化(Matlab代码实现)

&#x1f468;‍&#x1f393;个人主页&#xff1a;研学社的博客&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5;&#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密…

JVM知识整理

JVM知识整理 JVM的主要组成部分 JVM包含两个两个子系统&#xff08;类加载子系统和执行引擎&#xff09;和两个组件&#xff08;运行时数据区与和本地库接口&#xff09; 类加载子系统&#xff1a;根据给定的全限定类名来加载class文件到运行时数据区域中的方法区。执行引擎&a…

学大数据算跟风吗?

随着互联网、物联网和人工智能等技术的不断发展&#xff0c;大数据技术逐渐进入人们的视野&#xff0c;成为一个备受关注的热点话题。那么&#xff0c;大数据专业好学吗&#xff1f;前景如何&#xff1f;下面我们来一起探讨一下。 一、大数据专业的学习难度 大数据技术是一种综…

将 XLS 转换为 EXE:xlCompiler Crack

只需单击几下即可将Excel文件转换为应用程序 xl编译器无需编程即可将您的Excel电子表格转换为软件应用程序 将 XLS 转换为 EXE 将Excel文件转换为具有保护选项的应用程序。Excel 到 EXE 转换器为您提供了分发 Excel 模型的竞争优势和灵活性。将 Excel 的功能丰富的环境保存在应…

一文了解Gralde

&#x1f3e0;个人主页&#xff1a;shark-Gao &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是shark-Gao&#xff0c;一个想要与大家共同进步的男人&#x1f609;&#x1f609; &#x1f389;目前状况&#xff1a;23届毕业生&#xff0c;目前在某公司实习&#x1f…

蓝桥杯·3月份刷题集训Day02

本篇博客旨在记录自已打卡蓝桥杯3月份刷题集训&#xff0c;同时会有自己的思路及代码解答希望可以给小伙伴一些帮助。本人也是算法小白&#xff0c;水平有限&#xff0c;如果文章中有什么错误之处&#xff0c;希望小伙伴们可以在评论区指出来&#xff0c;共勉&#x1f4aa;。 文…

第14届蓝桥杯STEMA测评真题剖析-2023年3月12日Scratch编程初中级组

[导读]&#xff1a;超平老师的《Scratch蓝桥杯真题解析100讲》已经全部完成&#xff0c;后续会不定期解读蓝桥杯真题&#xff0c;这是Scratch蓝桥杯真题解析第113讲。 蓝桥杯选拔赛现已更名为STEMA&#xff0c;即STEM 能力测试&#xff0c;是蓝桥杯大赛组委会与美国普林斯顿多…

JavaScript 应用

目录 1、编程实现“计算任意区间内连续自然数的累加和”页面。 代码实现 2、应用 appendChild()方法和 getElementById()方法实现年月日的联动功能。 代码 1、编程实现“计算任意区间内连续自然数的累加和”页面。 &#xff08;1&#xff09;文档结构的创建 启动程序&#…

若依框架---权限管理设计

前言 若依权限管理包含两个部分&#xff1a;菜单权限 和 数据权限。菜单权限控制着我们可以执行哪些操作。数据权限控制着我们可以看到哪些数据。 菜单是一个概括性名称&#xff0c;可以细分为目录、菜单和按钮&#xff0c;以若依自身为例&#xff1a; 目录&#xff0c;就是页…

acm省赛:高桥和低桥(三种做法:区间计数、树状数组、线段树)

题目描述 有个脑筋急转弯是这样的&#xff1a;有距离很近的一高一低两座桥&#xff0c;两次洪水之后高桥被淹了两次&#xff0c;低桥却只被淹了一次&#xff0c;为什么&#xff1f;答案是&#xff1a;因为低桥太低了&#xff0c;第一次洪水退去之后水位依然在低桥之上&#xff…

Linux内核IO基础知识与概念

什么是 IO在计算机操作系统中&#xff0c;所谓的I/O就是 输入&#xff08;Input&#xff09;和输出&#xff08;Output&#xff09;&#xff0c;也可以理解为读&#xff08;Read&#xff09;和写&#xff08;Write)&#xff0c;针对不同的对象&#xff0c;I/O模式可以划分为磁盘…
最新文章