裤裤带你一起学C语言内存函数啦!

ops/2024/9/23 14:29:26/

目录

1.memcpy的使用和模拟实现

2.memmove使用和模拟实现

3.memset函数的使用

4.memcmp函数的使用


内存函数在<string.h>库中,我们使用内存函数必须先引入<string.h>头文件

1.memcpy的使用和模拟实现

memcpy的函数原型如下:

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

1.函数memcpy会从source的位置开始向后复制num个字节到destination中。

2.这个函数在遇到’\0‘时并不会停下来

3.如果两块内存有重叠的话,复制的结果是未定义的。

我们先来使用以下这个函数看看效果。

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

运行结果: 

1 2 0 0 0 0 0 0 0 0
1 2 3 0 0 0 0 0 0 0
1 2 3 4 5 0 0 0 0 0

大家一定要注意,这里操作的是字节,而一个字节是八个比特位。 

现在我们来模拟实现一下这个函数

void* memcpy(void* dst, const void* src, size_t count)
{assert(src);assert(dst);void* ret = dst;while (count--){*(char*)dst= *(char*)src;dst =(char*)dst + 1;src =(char*)src + 1;}return ret;
}

但是,如果两个内存块重叠的话,我们使用这个函数就无法完成我们的目的了,我给大家展示一下

为了避免出现这样的情况,我们就需要使用memmove来进行操作了。

2.memmove使用和模拟实现

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

那么我们又应该怎么去复制呢?诸君且看图!

void* memmove1(void* dst, const void* src, size_t count)
{void* ret = dst;//情况2,情况3if (dst <= ret || (char*)dst >= ((char*)src + count)){while (count--){*(char*)dst = *(char*)src;dst = (char*)dst + 1;src = (char*)src + 1;}}//别的情况都可以从高地址复制到低地址解决问题else{//我们如果不-1,就会比count多复制一个字节。dst = (char*)dst + count - 1;src = (char*)src + count - 1;while (count--){*(char*)dst = *(char*)src;dst = (char*)dst - 1;src = (char*)src - 1;}}return ret;
}

 再画个图帮助大家理解为什么要-1.

3.memset函数的使用

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

这个函数的原型如下:

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

ptr是要设置的内存地址

value是要设置的值

num是该值的字节数。

如下图所示:

4.memcmp函数的使用

这是一个内存比较函数,函数原型如下:

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

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

比较的原理和返回值可参考这篇博文目录6:C语言字符串函数全解!


http://www.ppmy.cn/ops/11415.html

相关文章

UE4_常见动画节点学习_Two Bone IK双骨骼IK

学习资料&#xff0c;仅供参考&#xff01; Two Bone IK 控制器将逆运动&#xff08;IK&#xff09;解算器应用于到如角色四肢等3关节链。 变量&#xff08; HandIKWeight &#xff09;被用于在角色的 hand_l 和 hand_r 控制器上驱动 关节目标位置&#xff08;Joint Target Lo…

2024-04-22(AJAX)

1.什么是Ajax 使用浏览器的XMLHttpRequest对象和服务器进行通信 浏览器网页中&#xff0c;使用Ajax技术&#xff08;XMLHttpRequest对象&#xff09;发起获取服务器数据的请求&#xff0c;服务器将数据给前端&#xff0c;前端拿到数据后&#xff0c;展示到网页。 2.为什么学…

milvus服务安装bash脚本指令理解

下拉镜像&#xff1a;docker pull milvusdb/milvus:v2.4.0-rc.1下载文件&#xff1a;https://hub.yzuu.cf/milvus-io/milvus/blob/master/scripts/standalone_embed.sh安装启动&#xff1a;bash standalone_embed.sh start详细解释下这段代码&#xff1a;wait_for_milvus_runni…

Yonbuilder参考

发布移动插件 https://developer.yonyou.com/cloud/moduleStore/publishPlugin Android自定义插件打包 社区问答 https://community.yonyou.com/forum.php?modviewthread&tid232830&searchLogId605932 MarkDown指令使用 https://blog.csdn.net/qq_25821067/article/de…

Xilinx FPGA BGA推荐设计规则和策略(二)

引言&#xff1a;上一篇介绍了BGA封装PCB层数估计、BGA焊盘设计、过孔设计、信号走线等内容&#xff0c;本文我们介绍下FPGA BGA封装电源管脚布线。 1. 概述 工程师必须在设计阶段早期评估功率需求&#xff0c;以确保有足够的层和面积为需要功率的BGA焊盘提供足够的功率。因为…

open-webui与ollama的部署最后完整之命令

docker run -d --networkhost -v open-webui:/app/backend/data -e HF_ENDPOINThttps://hf-mirror.com -e OLLAMA_BASE_URLhttp://127.0.0.1:11434 --name open-webui --restart always ghcr.io/open-webui/open-webui:main -e HF_ENDPOINThttps://hf-mirror.com 一定要加上&a…

FreeRTOS时间管理

FreeRTOS时间管理 主要要了解延时函数&#xff1a; 相对延时&#xff1a;指每次延时都是从执行函数vTaskDelay()开始&#xff0c;直到延时指定的时间结束。 绝对延时&#xff1a;指将整个任务的运行周期看成一个整体&#xff0c;适用于需要按照一定频率运行的任务。 函数 vTa…

canvas 学习

最近的项目涉及到 canvas 相关的知识&#xff0c;就在网站上找资源先大概了解一下&#xff0c;然后再细细研究。 看到了一篇 “canvas详细教程” 的资源&#xff0c;感觉十分不错&#xff0c;就分享给大家&#xff1a; canvas详细教程! ( 近1万字吐血总结)这期是潘潘整理的万…