当前位置: 首页 > news >正文

C语言之内存函数

C语言之内存函数

文章目录

  • C语言之内存函数
    • 1. memcpy 使⽤和模拟实现
      • 1.1 memcpy 函数的使用
      • 1.3 memcpy的模拟实现
    • 2. memmove 使⽤和模拟实现
      • 2.1 memmove 函数的使用
      • 2.2 memmove的模拟实现
    • 3. memset 函数的使用
    • 4. memcmp 函数的使⽤

1. memcpy 使⽤和模拟实现

函数声明如下:

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

memcpy函数和strncpy函数很相似,

  1. 只不过strncpy函数只能处理字符数组,而memcpy函数可以处理任意类型的数据
  2. strncpy函数会在目标字符数组结尾添加’\0’,而memcpy函数则不会
  3. 都在string.h库函数中

1.1 memcpy 函数的使用

代码一:
将arr1中的1 2 3 4 5 拷贝到arr2中

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

代码运行结果如下:
在这里插入图片描述

memcpy函数有三个参数

  1. 参数一为需要拷贝的目标地址
  2. 参数二为拷贝数据来源的地址
  3. 参数三为需要拷贝的字节大小
    在上述代码中,传给了memcpy函数20个字节的大小
    因为整型是4个字节,所以只会拷贝前5个数据

代码二:
将arr1中的4 5 6 7 8拷贝到arr2中

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

在这里插入图片描述

代码三:
将arr1中的字符串拷贝到arr2中

#include <stdio.h>
#include <string.h>
int main()
{char arr1[] = "Hello \0World!";char arr2[20] = { 0 };memcpy(arr2, arr1, 13);int i = 0;for (i = 0; i < 13; i++){printf("%c", arr2[i]);}return 0;
}

代码运行结果如下:
在这里插入图片描述
memcpy函数在拷贝数据时,遇到\0时不会停下来

代码四:
将arr中的 1 2 3 4 5 拷贝到arr中4 5 6 7 8的位置上

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

在这里插入图片描述

由于空间重叠了,当改dest的第三个元素时,src的第三个元素已经被改成了1,使用dest第三个元素为1,第四个同理,为2,所以使用memcpy无法拷贝重复的空间

在VS2022中可以实现
在这里插入图片描述
但是
C语言中规定:
memcpy拷贝的就是不重复的数据
重复的数据交给memmove

在VS2022中,memcpy是可以拷贝重复部分的数据的,在C语言中规定的memcpy功能可能只有60分,但是VS2022中可能将其提升到了100分

总结:
• 函数memcpy从source的位置开始向后复制num个字节的数据到destination指向的内存位置。
• 这个函数在遇到 ‘\0’ 的时候并不会停下来。
• 如果source和destination有任何的重叠,复制的结果都是未定义的

1.3 memcpy的模拟实现

void *memcpy(void *dest, const void *src, size_t n);

思路:

  1. 由于memcpy要处理不同类型的数据,所以第一个参数和第二个参数得是void类型的,void类型的指针可以接收任意类型的地址
  2. 由于不确定传进来的是什么数据,只知道需要拷贝多少字节,在函数内部可以将其强制类型转换为char*类型的指针,一个字节一个字节拷贝
  3. 函数返回值为dest的地址,所以为void*

实现代码如下:

#include <stdio.h>
#include <assert.h>void* my_memcpy(void* dest, const void* src, size_t num)
{char* ret = dest;assert(dest && src); //assert断言,判断传入的地址是否为空指针while (num--)   //一个字节一个字节拷贝{*(char*)dest = *(char*)src; //void*类型不能解引用,并且需要一个字节一个字节的拷贝,所以强制类型转换为char*类型dest = (char*)dest + 1; //强制类型转换是临时的,再次强制类型转换并+1找到下一个需要交换的地址src = (char*)src + 1;}return ret;  //返回dest的地址
}int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };int arr2[20] = { 0 };my_memcpy(arr2, arr1, 5 * sizeof(int));int i = 0;for (i = 0; i < 10; i++){printf("%d ", arr2[i]);}return 0;

2. memmove 使⽤和模拟实现

函数声明如下:

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

• 和memcpy的差别就是memmove函数处理的源内存块和⽬标内存块是可以重叠的

• 如果源空间和⽬标空间出现重叠,就得使⽤memmove函数处理。

2.1 memmove 函数的使用

将arr中的 1 2 3 4 5 拷贝到arr中4 5 6 7 8的位置上

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

代码运行结果如下:
在这里插入图片描述

2.2 memmove的模拟实现

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

思路:

  1. 由于memmove也要处理不同类型的数据,所以第一个参数和第二个参数得是void类型的,void类型的指针可以接收任意类型的地址
  2. 由于不确定传进来的是什么数据,只知道需要拷贝多少字节,在函数内部可以将其强制类型转换为char*类型的指针,一个字节一个字节拷贝
  3. 函数返回值为dest的地址,所以为void*
  4. dest 和 src 会有两种关系
    1.dest 在 src 左边
    2.dest 在 src 右边

情况一:
在这里插入图片描述

上述情况,虽然有重叠的空间,但是还是可以拷贝,实现方式和memcpy一样

情况二:

在这里插入图片描述

在上述代码中,出现了重叠的空间,如果正序交换的话,后面的内容被覆盖了,无法交换,所以我们换种方式交换,逆序交换
在这里插入图片描述
这样交换,内容就不会被覆盖了

实现代码如下:

#include <stdio.h>
#include <assert.h>
void* my_memcpy(char* dest, const char* src, size_t num)
{char* ret = dest; assert(dest && src);   //assert断言if (dest < src)   //当dest在src左边时{while(num--)      //逻辑和memcpy一致{*(char*)dest = *(char*)src;dest = (char*)dest + 1;src = (char*)src + 1;}}else             //dest在src右边{while (num--)  //一个字节一个字节拷贝{*((char*)dest + num) = *((char*)src + num); //找到最后一个字节然后拷贝 num--之后找到倒数第二个字节拷贝}}
}
int main()
{int arr[] = { 1,2,3,4,5,6,7,8,9,10 };my_memcpy(arr + 3, arr, 5 * sizeof(int));int i = 0;for (i = 0; i < 10; i++){printf("%d ", arr[i]);}return 0;
}

3. memset 函数的使用

函数声明如下:

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

memset是⽤来设置内存的,将内存中的值以字节为单位设置成想要的内容
第一个参数是需要修改的地址
第二个参数是设置成什么内容
第三个参数是要设置的大小

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

代码运行结果:
xxxxxfghi

将前5个字节的元素改成了x

4. memcmp 函数的使⽤

函数声明如下:

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

和strncmp相似
都是⽐较从ptr1和ptr2指针指向的位置开始,向后的num个字节
返回值如下:
如果ptr1大于ptr2则返回一个大于零的值
如果ptr1小于ptr2则返回一个小于零的值
如果ptr1等于ptr2则返回一个零
在这里插入图片描述

#include <stdio.h>
#include <string.h>
int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };int arr2[] = { 1,2,3,4,9 };int ret = memcmp(arr1, arr2, 16);if (ret > 0)printf("大于\n");else if (ret < 0)printf("小于");elseprintf("等于\n");return 0;
}

代码运行结果:
等于

相关文章:

C语言之内存函数

C语言之内存函数 文章目录 C语言之内存函数1. memcpy 使⽤和模拟实现1.1 memcpy 函数的使用1.3 memcpy的模拟实现 2. memmove 使⽤和模拟实现2.1 memmove 函数的使用2.2 memmove的模拟实现 3. memset 函数的使用4. memcmp 函数的使⽤ 1. memcpy 使⽤和模拟实现 函数声明如下&a…...

基本数据结构二叉树(1)

目录 1.树概念及结构 1.1树的概念 1.2 树的相关概念 1.3 树的表示 1.4 树在实际中的运用&#xff08;表示文件系统的目录树结构&#xff09; 2.二叉树概念及结构 2.1概念 2.2现实中的二叉树&#xff1a; 2.3 特殊的二叉树&#xff1a; 2.5 二叉树的存储结构 2. 链式存…...

【python】Python将100个PDF文件对应的json文件存储到MySql数据库(源码)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;公众号&#x1f448;&#xff1a;测试开发自动化【获取源码商业合作】 &#x1f449;荣__誉&#x1f448;&#xff1a;阿里云博客专家博主、5…...

Android:Google三方库之Adjust集成详细步骤

通过 Adjust 安卓 SDK&#xff0c;您可以在自己的安卓应用中跟踪归因、事件及更多数据。请按照本指南中说明的步骤操作&#xff0c;在应用内设置 Adjust SDK 1、添加依赖 //adjustimplementation("com.adjust.sdk:adjust-android:4.33.5")implementation("com.…...

prometheus|云原生|grafana-9.4.3版本的主题更改

一&#xff0c; grafana-9.4.3版本的主题更改 grafana-9.4.3版本应该是目前比较高的版本了&#xff0c;但不知道是什么原因&#xff0c;grafana的主题界面并不多&#xff0c;只有暗色&#xff0c;亮色和系统色三种 配置管理----首选项里可以看到 亮色&#xff1a; 暗色&…...

B033-Servlet交互 JSP

目录 ServletServlet的三大职责跳转&#xff1a;请求转发和重定向请求转发重定向汇总请求转发与重定向的区别用请求转发和重定向完善登录 JSP第一个JSP概述注释设置创建JSP文件默认字符编码集 JSP的java代码书写JSP的原理三大指令九大内置对象改造动态web工程进行示例内置对象名…...

Less 安装教程

文章目录 前言LESS的系统要求安装LESS例子输出Less编译css工具后言 前言 hello world欢迎来到前端的新世界 &#x1f61c;当前文章系列专栏&#xff1a;Sass和Less &#x1f431;‍&#x1f453;博主在前端领域还有很多知识和技术需要掌握&#xff0c;正在不断努力填补技术短板…...

Java研学-多线程

一 名词解析 1 线程 : 控制指定APP(进程)执行的最基本单元(最小单位) 2 进程 : 硬件设备上的每一个应用程序 3 单线程 : 一个进程中只有一个线程执行,实际上基本没有这种情况 4 多线程 : 一个进程中至少有两个或两个以上的线程在执行 二 创建方式 1 共有三种:Thread类. R…...

【日常总结】如何禁止浏览器 http自动跳转成https

一、场景 二、问题 三、解决方案 3.1 chrome 浏览器 3.2 edge 浏览器&#xff1a; 3.3 Safari 浏览器 3.4 Firefox 浏览器 3.5 Microsoft Edge 一、场景 公司网站 http:// 谷歌浏览器中自动转换成 https:// 导致无法访问 二、问题 nginx配置ssl 443接口&#xff0c; ht…...

文本转语音:微软语音合成标记语言 (SSML) 文本结构和事件

​ SSML 的语音服务实现基于万维网联合会的语音合成标记语言版本 1.0。 ​ 语音服务支持的元素可能与 W3C 标准不同。 每个 SSML 文档是使用 SSML 元素&#xff08;或标记&#xff09;创建的。 这些元素用于调整语音、风格、音节、韵律、音量等。 下面是 SSML 文档的基本结构…...

计算机网络之物理层(数据通信有关)

一、概述 1.1物理层引入的目的 屏蔽掉传输介质的多样性&#xff0c;导致数据传输方式的不同&#xff1b;物理层的引入使得高层看到的数据都是统一的0,1构成的比特流 1.2.物理层如何实现屏蔽 物理层靠定义的不同的通信协议&#xff08;一般称通信规程&#xff09; 这些协议…...

安卓开发之HTTP API服务接口设计(基于okhttp3请求)

安卓中的请求 OkHttp3 是一个开源的 Java/Android HTTP 客户端库,由 Square 公司开发。它提供了简洁和高效 的 API ,用于进行 HTTP 请求、处理响应以及与服务器进行通信。 以下是 OkHttp3 的一些主要特点和功能: 简单易用: OkHttp3 提供了简洁的 API ,使得发送 HTTP 请求变…...

我的第一次SACC之旅

今年有很多第一次&#xff0c;第一次作为“游客”参加DTCC&#xff08;中国数据库大会&#xff09;&#xff0c;第一次作为讲师参与ACDU中国行&#xff08;成都站&#xff09;&#xff0c;第一次参加OB年度发布会&#xff08;包含DBA老友会&#xff09;&#xff0c;而这次是第一…...

Codeforces Round 908 (Div. 2)

Codeforces Round 908 (Div. 2) 文章目录 Codeforces Round 908 (Div. 2)ABCD A 获取对战的最后一个即为胜者 #include <bits/stdc.h>using namespace std;void solve() {int n;cin >> n;string s;cin >> s;cout << s[n-1] << endl; }int main…...

Java实现-数据结构 2.时间和空间复杂度

.如何衡量一个算法的好坏&#xff1a;时间复杂度和空间复杂度 算法效率分为时间效率和空间效率&#xff0c;时间效率称为时间复杂度&#xff0c;空间效率称为空间复杂度 时间复杂度 算法的时间复杂度是一个数学函数&#xff0c;它描述了算法的运行时间&#xff0c;一个算法执…...

Docker exec命令

docker exec &#xff1a;在运行的容器中执行命令。 语法&#xff1a; docker exec [OPTIONS] CONTAINER COMMAND [ARG...]OPTIONS说明&#xff1a; -d&#xff1a;分离模式&#xff1a; 在后台运行 -i&#xff1a;即使没有附加也保持STDIN打开 -t&#xff1a;分配一个伪终…...

可燃气体监测仪助力燃气管网安全监测,效果一览

城市地下管线是指城市范围内供应水、排放水、燃气等各类管线及其附属设施&#xff0c;它们是保障城市正常运转的重要基础设施且影响着城市生命线。其中燃气引发的事故近些年不断增加&#xff0c;由于燃气管线深埋地下环境复杂&#xff0c;所以仅仅依赖人工巡查难以全面有效地防…...

Kafka(二)在WSL搭建Schema Registry

目录 1 Avro与Schema Registry2 搭建Schema Registry2.1 下载Confluent并解压2.2 设置环境变量2.3 修改配置2.4 启动服务 3 API列表 1 Avro与Schema Registry Apache Avro 是一种高效的数据序列化系统&#xff0c;用于在不同的应用程序和平台之间传输和存储数据。它提供了一种…...

webrtc AEC 线性滤波 PBFDAF(均匀分块频域自适应滤波)介绍

计算一个脉冲响应和输入信号的卷积&#xff0c;除了使用原始的时域卷积以外&#xff0c;还有如下方法&#xff1a; FFT卷积的方法&#xff1a;对输入信号&#xff08;长度M&#xff09;和脉冲响应&#xff08;长度N&#xff09;分别补零到K&#xff08;K>MN-1)&#xff0c;…...

开源vs闭源,处在大模型洪流中,向何处去?

文章目录 一、开源和闭源的优劣势比较1.1 开源优势1.2 闭源的优势 二、开源和闭源对大模型技术发展的影响2.1 数据共享2.2 算法创新2.3 业务拓展2.4 安全性和隐私2.5 社会责任和伦理 三、开源与闭源的商业模式比较3.1 盈利模式3.2 市场竞争3.3 用户生态3.4 创新速度 四&#xf…...

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…...

挑战杯推荐项目

“人工智能”创意赛 - 智能艺术创作助手&#xff1a;借助大模型技术&#xff0c;开发能根据用户输入的主题、风格等要求&#xff0c;生成绘画、音乐、文学作品等多种形式艺术创作灵感或初稿的应用&#xff0c;帮助艺术家和创意爱好者激发创意、提高创作效率。 ​ - 个性化梦境…...

【Python】 -- 趣味代码 - 小恐龙游戏

文章目录 文章目录 00 小恐龙游戏程序设计框架代码结构和功能游戏流程总结01 小恐龙游戏程序设计02 百度网盘地址00 小恐龙游戏程序设计框架 这段代码是一个基于 Pygame 的简易跑酷游戏的完整实现,玩家控制一个角色(龙)躲避障碍物(仙人掌和乌鸦)。以下是代码的详细介绍:…...

Java - Mysql数据类型对应

Mysql数据类型java数据类型备注整型INT/INTEGERint / java.lang.Integer–BIGINTlong/java.lang.Long–––浮点型FLOATfloat/java.lang.FloatDOUBLEdouble/java.lang.Double–DECIMAL/NUMERICjava.math.BigDecimal字符串型CHARjava.lang.String固定长度字符串VARCHARjava.lang…...

SpringBoot+uniapp 的 Champion 俱乐部微信小程序设计与实现,论文初版实现

摘要 本论文旨在设计并实现基于 SpringBoot 和 uniapp 的 Champion 俱乐部微信小程序&#xff0c;以满足俱乐部线上活动推广、会员管理、社交互动等需求。通过 SpringBoot 搭建后端服务&#xff0c;提供稳定高效的数据处理与业务逻辑支持&#xff1b;利用 uniapp 实现跨平台前…...

学习STC51单片机32(芯片为STC89C52RCRC)OLED显示屏2

每日一言 今天的每一份坚持&#xff0c;都是在为未来积攒底气。 案例&#xff1a;OLED显示一个A 这边观察到一个点&#xff0c;怎么雪花了就是都是乱七八糟的占满了屏幕。。 解释 &#xff1a; 如果代码里信号切换太快&#xff08;比如 SDA 刚变&#xff0c;SCL 立刻变&#…...

Maven 概述、安装、配置、仓库、私服详解

目录 1、Maven 概述 1.1 Maven 的定义 1.2 Maven 解决的问题 1.3 Maven 的核心特性与优势 2、Maven 安装 2.1 下载 Maven 2.2 安装配置 Maven 2.3 测试安装 2.4 修改 Maven 本地仓库的默认路径 3、Maven 配置 3.1 配置本地仓库 3.2 配置 JDK 3.3 IDEA 配置本地 Ma…...

力扣-35.搜索插入位置

题目描述 给定一个排序数组和一个目标值&#xff0c;在数组中找到目标值&#xff0c;并返回其索引。如果目标值不存在于数组中&#xff0c;返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 class Solution {public int searchInsert(int[] nums, …...

Java求职者面试指南:Spring、Spring Boot、MyBatis框架与计算机基础问题解析

Java求职者面试指南&#xff1a;Spring、Spring Boot、MyBatis框架与计算机基础问题解析 一、第一轮提问&#xff08;基础概念问题&#xff09; 1. 请解释Spring框架的核心容器是什么&#xff1f;它在Spring中起到什么作用&#xff1f; Spring框架的核心容器是IoC容器&#…...

MFC 抛体运动模拟:常见问题解决与界面美化

在 MFC 中开发抛体运动模拟程序时,我们常遇到 轨迹残留、无效刷新、视觉单调、物理逻辑瑕疵 等问题。本文将针对这些痛点,详细解析原因并提供解决方案,同时兼顾界面美化,让模拟效果更专业、更高效。 问题一:历史轨迹与小球残影残留 现象 小球运动后,历史位置的 “残影”…...