【C语言】memcpy , memset等内存操作函数使用方法与注意事项

这个章节,我们探讨C语言内存操作函数。
重点介绍处理内存操作函数使用和注意事项
和内存函数如何模拟实现。

内存函数所需头文件
#include<string.h>

文章目录

  • memcpy
    • memcpy 函数模拟实现
  • memmove
    • memmove 函数模拟实现
  • memcmp
    • memcmp 函数模拟实现
  • memset
    • memset 函数模拟实现

memcpy

从source的位置开始向后复制num个字节的数据到destination的内存位置

void * memcpy ( void * destination, const void * source, size_t num );
参数说明:

destination: 指向用于存储复制内容的目标数组 。

source: 指向将要复制的数据。

num: 复制的字节个数

返回值:

返回一个指向 destination 首元素地址的指针

注意事项:

  1. 这个函数在遇到 ‘\0’ 的时候并不会停下来。
  2. 如果source和destination有任何的重叠,复制的结果都是未定义的。
  3. 只要处理 不重叠的内存拷贝就可以

函数使用方法:

#include<stdio.h>
#include<string.h>
typedef struct Stu
{
	char nane[20];
	int age;
}T;
int main()
{
	T t[3] = { {"张三",20},{"李四",30},{"小刘",40} };
	T s[4] = { 0 };
	memcpy(s, t, sizeof(t));//把t[3]的内容 拷贝到s里面
	int i = 0;
	for (i = 0; i < 3; i++)
	{
		printf("%s %d\n", s[i].nane, s[i].age);
	}
	return 0;
}

最终的输出结果:
张三 20
李四 30
小刘 40

memcpy 函数模拟实现

比较num个字节,把他转换成(char*)每次+1转换一个字节,直到转换num个字节。

#include<stdio.h>
#include<string.h>
void* my_memcpy(void* arr1, void* arr2, int num)
{
   assert(arr1 && arr2);
   void* ret = arr1;
   while (num--)
    {
      *((char*)arr1) = *((char*)arr2);
      (char*)arr1 = (char*)arr1 + 1;
      (char*)arr2 = (char*)arr2 + 1;
    }
    return ret;
}

memmove

memmove 能够保证源串在被覆盖之前将重叠区域的字节拷贝到目标区域中,复制后源区域的内容会被更改。如果目标区域与源区域没有重叠,则和 memcpy() 函数功能相同

void * memmove ( void * destination, const void * source, size_t num );
参数说明:

  1. 指向用于存储复制内容的目标数组 。
  2. source: 指向将要复制的数据
  3. num: 复制的字节个数

返回值:

返回一个指向 destination 首元素地址的指针

注意事项:

  1. 和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的
  2. 如果源空间和目标空间出现重叠,就得使用memmove函数处理。

函数的使用方法:

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

最终的输出结果:
1 2 1 2 3 4 5 8 9

memmove 函数模拟实现

memmove函数的实现分两个部分,因为存在内存覆盖的情况,如果arr2首元素地址大于arr1首元素地址,那肯定需要从前往后拷贝,反之后往前即可。

void* my_memmove(void* arr1, void* arr2, int num)
{
	assert(arr1 && arr2);
	char* ret = arr1;
	if (arr1 < arr2)
	{
		while (num--)
		{
			//从前往后
			*((char*)arr1) = *((char*)arr2);
			(char*)arr1 = (char*)arr1 + 1;
			(char*)arr2 = (char*)arr2 + 1;
		}
	}
	else
	{
		//从后往前
		while (num--)
		{
			*((char*)arr1 + num) = *((char*)arr2 + num);
		}
	}
	return ret;
}

memcmp

内存存储的字节 str1 和 内存存储的字节 str2 的前 num 个字节进行比较

int memcmp(const void *str1, const void *str2, size_t num)
参数说明:

str1: 指向str1内存块的指针
str2: 指向str2内存块的指针
num: 要比较字节的个数

返回值:

  1. str1 大于 str2,则返回大于0的数字
  2. str1 等于str2 ,则返回0
  3. str1 小于 str2,则返回小于0的数字

函数的使用方法:

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

memcmp 函数模拟实现

int my_memcmp(void* arr1, void* arr2, int num)
{
	assert(arr1 && arr2);
	while (*((char*)arr1) == *((char*)arr2) && num--)
	{
		(char*)arr1 = (char*)arr1 + 1;
		(char*)arr2 = (char*)arr2 + 1;
	}
	if (*((char*)arr1) > *((char*)arr2))
	{
		return 1;
	}
	else if (*((char*)arr1) < *((char*)arr2))
	{
		return -1;
	}
	else
	{
		return 0;
	}
}

memset

复制字符 c(一个无符号字符)到参数 str 所指向地址。复制num个字节

void *memset(void *str, int c, size_t num)
参数说明:

str: 指向要复制的的内存块地址
c: 是要被设置的值
num: 要复制多少个字节的个数

返回值:

返回指向srr首元素地址的指针

注意事项:

  1. 不能任意赋值,memset函数是按照字节对内存块进行初始化,如果是int类型的数组,只能初始化0或者-1.
  2. 注意memset,是按字节来复制的,而不是按元素个数,经常有人在这里出错。

函数的使用方法:

#include<stdio.h>
#include<string.h>
int main()
{
	char arr[] = "abcdef";
	memset(arr, '#', 4);
	printf("%s\n", arr);
	return 0;
}

memset 函数模拟实现

void* my_memset(void* arr, int set, int num)
{
assert(arr);
void* ret = arr;
while (num--)
{
*((char*)arr) = (char)set;
(char*)arr = (char*)arr + 1;
}
return ret;
}

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

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

相关文章

python并发编程多线程

在传统操作系统中&#xff0c;每个进程有一个地址空间&#xff0c;而且默认就有一个控制线程 线程顾名思义&#xff0c;就是一条流水线工作的过程&#xff0c;一条流水线必须属于一个车间&#xff0c;一个车间的工作过程是一个进程 车间负责把资源整合到一起&#xff0c;是一个…

C语言指针操作(十)动态内存分配与指向它的指针变量

目录 一、什么是内存的动态分配 二、怎样建立内存的动态分配 2.1用malloc函数开辟动态存储区 2.2用calloc函数开辟动态存储区 2.3用realloc函数重新分配动态存储区 2.4用free函数释放动态存储区 三、void指针类型 四、举例说明 一、什么是内存的动态分配 全局变量是分…

redis持久化的几种方式

一、简介 Redis是一种高级key-value数据库。它跟memcached类似&#xff0c;不过数据可以持久化&#xff0c;而且支持的数据类型很丰富。有字符串&#xff0c;链表&#xff0c;集 合和有序集合。支持在服务器端计算集合的并&#xff0c;交和补集(difference)等&#xff0c;还支持…

【蓝桥杯专题】 贪心(C++ | 洛谷 | acwing | 蓝桥)

菜狗现在才开始备战蓝桥杯QAQ 文章目录【蓝桥杯专题】 &#xff08;C | 洛谷 | acwing | 蓝桥&#xff09;1055. 股票买卖 IIAcWing 104. 货仓选址传递糖果AcWing 112. 雷达设备付账问题乘积最大AcWing 1247. 后缀表达式P【蓝桥杯专题】 &#xff08;C | 洛谷 | acwing | 蓝桥&…

Flink 应用案例——求网页访问量Top N 实时计算(附可执行代码)

在学习了Flink之后&#xff0c;笔者通过以下案例对Flink API 进行简单复习 目录 案例要求 前置准备 编写主程序&#xff08;点此跳转至代码&#xff09; 运行截图 案例要求 以下数据 为某网站的访问日志 现要求通过以下数据 统计出最近10s内最热门的N个页面&#xff08;即…

【3.17】MySQL索引整理、回溯(分割、子集问题)

3.1 索引常见面试题 索引的分类 什么是索引&#xff1f; 索引是一种数据结构&#xff0c;可以帮助MySQL快速定位到表中的数据。使用索引&#xff0c;可以大大提高查询的性能。 按「数据结构」分类&#xff1a;Btree索引、Hash索引、Full-text索引。 InnoDB 存储引擎创建的聚簇…

漫画:什么是快速排序算法?

这篇文章&#xff0c;以对话的方式&#xff0c;详细着讲解了快速排序以及排序排序的一些优化。 一禅&#xff1a;归并排序是一种基于分治思想的排序&#xff0c;处理的时候可以采取递归的方式来处理子问题。我弄个例子吧&#xff0c;好理解点。例如对于这个数组arr[] { 4&…

优思学院|六西格玛DMAIC,傻傻搞不清?

DMAIC还是搞不清&#xff1f; DMAIC是一个用于过程改进和六西格玛的问题解决方法论。它是以下五个步骤的缩写&#xff1a; 定义&#xff08;Define&#xff09;&#xff1a;明确问题&#xff0c;设定项目的目标和目的。绘制流程图&#xff0c;并收集数据&#xff0c;以建立未来…

基于bearpi的智能小车--Qt上位机设计

基于bearpi的智能小车--Qt上位机设计 前言一、界面原型1.主界面2.网络配置子窗口模块二、设计步骤1.界面原型设计2.控件添加信号槽3.源码解析3.1.网络链接核心代码3.2.网络设置子界面3.3.小车控制核心代码总结前言 最近入手了两块小熊派开发板,借智能小车案例,进行鸿蒙设备学…

01背包问题c++

问题 问题介绍 有 N 种物品和一个容量是 V 的背包&#xff0c;每种物品都有无限件可用。 第 i 种物品的体积是 vi&#xff0c;价值是 wi。 求解将哪些物品装入背包&#xff0c;可使这些物品的总体积不超过背包容量&#xff0c;且总价值最大。 输出最大价值。 输入格式 第…

基于Transformer的交通预测模型部分汇总【附源代码】

交通预测一直是一个重要的问题&#xff0c;它涉及到交通运输系统的可靠性和效率。随着人工智能的发展&#xff0c;越来越多的研究者开始使用深度学习模型来解决这个问题。其中&#xff0c;基于Transformer的交通预测模型在近年来备受关注&#xff0c;因为它们具有优秀的建模能力…

设计模式之桥接模式(C++)

作者&#xff1a;翟天保Steven 版权声明&#xff1a;著作权归作者所有&#xff0c;商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处 一、桥接模式是什么&#xff1f; 桥接模式是一种结构型的软件设计模式&#xff0c;将抽象部分与实现部分分离&#xff0c;使他们可…

像ChatGPT玩转Excel数据

1.引言 最近ChatGPT的出现&#xff0c;把人工智能又带起了一波浪潮。机器人能否替代人类又成了最近热门的话题。 今天我们推荐的一个玩法和ChatGPT有点不一样。我们的课题是“让用户可以使用自然语言从Excel查询到自己想要的数据”。 要让自然语言可以从Excel中查数据&#…

通过百度文心一言大模型作画尝鲜,感受国产ChatGPT的“狂飙”

3月16日下午&#xff0c;百度于北京总部召开新闻发布会&#xff0c;主题围绕新一代大语言模型、生成式AI产品文心一言。百度创始人、董事长兼首席执行官李彦宏&#xff0c;百度首席技术官王海峰出席&#xff0c;并展示了文心一言在文学创作、商业文案创作、数理推算、中文理解、…

用Qt画一个温度计

示例1 以下是用Qt绘制一个简单的温度计的示例代码&#xff1a; #include <QPainter> #include <QWidget> #include <QApplication> class Thermometer : public QWidget { public:Thermometer(QWidget *parent 0); protected:void paintEvent(QPaintEvent …

【Hive】配置

目录 Hive参数配置方式 参数的配置方式 1. 文件配置 2. 命令行参数配置 3. 参数声明配置 配置源数据库 配置元数据到MySQL 查看MySQL中的元数据 Hive服务部署 hiveserver2服务 介绍 部署 启动 远程连接 1. 使用命令行客户端beeline进行远程访问 metastore服务 …

LC-146.LRU 缓存

题解&#xff1a;https://leetcode.cn/problems/lru-cache/solution/lru-ce-lue-xiang-jie-he-shi-xian-by-labuladong/ 文章目录[146. LRU 缓存](https://leetcode.cn/problems/lru-cache/)思路从0开始实现使用LinkedHashMap实现拓展&#xff1a;[460. LFU 缓存](https://leet…

【2024考研】计算机考研,4轮复习时间安排

文章目录&#x1f3a8;第1轮复习&#xff08;暑假前&系统课&#xff09;英语1/2数学1/2专业课408&#x1f3a8;第2轮复习&#xff08;开学前&真题&#xff09;英语1/2试卷数学1/2试卷专业课408试卷&#x1f3a8;第3轮复习&#xff08;报名前&政治&#xff09;政治试…

什么是数据治理,如何保障数据质量?_光点科技

随着信息化和数据化的发展&#xff0c;数据已经成为企业最为重要的资产之一。数据治理作为一种管理和保障数据质量的方法&#xff0c;越来越受到企业的重视。什么是数据治理&#xff1f;数据治理是一种管理和保障数据质量的方法。数据治理的主要目的是确保数据的可靠性、准确性…

Android APP隐私合规检测工具Camille使用

目录一、简介二、环境准备常用使用方法一、简介 现如今APP隐私合规十分重要&#xff0c;各监管部门不断开展APP专项治理工作及核查通报&#xff0c;不合规的APP通知整改或直接下架。camille可以hook住Android敏感接口&#xff0c;检测是否第三方SDK调用。根据隐私合规的场景&a…
最新文章