C语言学习系列-->一篇带你看懂内存函数

这篇具有很好参考价值的文章主要介绍了C语言学习系列-->一篇带你看懂内存函数。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。


C语言学习系列-->一篇带你看懂内存函数,c语言,学习,开发语言

前言

上篇文章学习了C语言字符串函数,只是对字符串进行操作
本节,小编整理了一下C语言中的内存函数,对内存进行操作,只针对会内存块,不针对数据

memcpy

概述

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

memcpy是对内存拷贝

拷贝的可能是字符串,也可能是整型数组

所以使用 void*

将source拷贝到destination,指定字节数为num

code

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

arr2是dest,arr1是scr,20是怒num,即字节

将arr1中的前20个字节(即1,2,3,4,5)复制到arr2中

运行结果

1 2 3 4 5 0 0 0 0 0

模拟实现

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

void* my_memcpy(void* dest,const void* src, size_t num)
{
	void* ret = dest;    //后面dest的地址可能会发生改变,因此先将dest的首地址存放在ret中
	assert(dest && src);
	while (num--)
	{
		*(char*) dest = *(char*)src;   //viod*型不能解引用,所以先强制类型转换成char*型,再解引用
		dest = (char*)dest+1;
		src = (char*)src+1;
	}
}

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;
}

对于重叠的内存,交给memmove来处理

其实在vs中,memcpy库函数也可以处理重叠内存,但是在其他编译器不一定可以,不通用

memmove

概述

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

• 和memcpy的差别就是memmove函数处理的源内存块和⽬标内存块是可以重叠的。
• 如果源空间和⽬标空间出现重叠,就得使⽤memmove函数处理。

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

即,最终结果为,1 2 1 2 3 4 5 8 9 10
C语言学习系列-->一篇带你看懂内存函数,c语言,学习,开发语言

code

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

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

输出结果

1 2 1 2 3 4 5 8 9 10

模拟实现

前面我们看到,

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

C语言学习系列-->一篇带你看懂内存函数,c语言,学习,开发语言

实际上应该是

将5放到7的位置
将4放到6的位置
将3放到5的位置
将4放到4的位置
将3放到3的位置

是后—>前

如果,先将1放到3的位置,那么3这个位置就变成1。再将2放到4的位置,此时最终结果就变成了1 2 1 2 1 2 1 8 9 10

再来看看这种情况:

C语言学习系列-->一篇带你看懂内存函数,c语言,学习,开发语言
将5 6 7 8 9拷贝到3 4 5 6 7的位置上

将5放到3的位置上
将6放到4的位置上
(此时原本5 6位置上就没有数字了)
将7放到5的位置
将8放到6的位置
将9放到7的位置

是前—>后

C语言学习系列-->一篇带你看懂内存函数,c语言,学习,开发语言
因此,
if(dest<src):前—>后
else:后—>前

模拟code

#include <stdio.h>
#include <string.h>
#include<assert.h>
void* my_memmove(void* dest, const void* src, size_t num)
{
	void* ret = dest;
	assert(dest);
	assert(src);

	//分情况讨论
	if (dest < src)
	{
		//前->后
		while (num--)
		{
			*(char*)dest = *(char*)src;
			dest = (char*)dest + 1;
			src = (char*)src + 1;
		}
	}
	else
	{
		//后->前
		while (num--)
		{
			//num=18
			*((char*)dest + num) = *((char*)src + num);
		}
	}
	return ret;
}

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

	return 0;
}

输出结果

3 4 5 6 7 6 7 8 9 10

memset

memory set 记忆设置,即内存设置

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

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

code

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

注意点:memset是以字节为单位设置内存值的

输出结果

xxxxxxworld

memcmp

和strcmp功能其实差不多

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

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

code

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[] = "abcdef";
	char arr2[] = "abqwertyuiop";
	int ret = memcmp(arr1, arr2, 2);
	printf("%d\n", ret);

	return 0;
}

输出结果

0

总结

小编看来,内存函数和字符串函数有很多相似的地方,只不过内存函数针对的内存块罢了。
下一篇更新:数据在内存中的储存

C语言学习系列-->一篇带你看懂内存函数,c语言,学习,开发语言文章来源地址https://www.toymoban.com/news/detail-708157.html

到了这里,关于C语言学习系列-->一篇带你看懂内存函数的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • GRBL源码详解步进电机的算法学习笔记(STM32)一篇带你学完GRBL算法初版

    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 GRBL源码中步进电机的算法学习笔记(STM32) GRBL源码中算法部分的学习是我在公司研发激光切割机(三轴)期间研究的一套关于步进电机驱动控制的常见算法,以下内容都以激光切割机来举例,话不多说

    2024年02月04日
    浏览(53)
  • 一篇带你精通MPLS

    MPLS:多协议标签交换 可以基于多种不同的3层协议来生成2.5层的标签信息 包为网络层的PDU,故包交换就是基于IP地址进行数据转发;也就是路由器的路由行为。(路由器和终端基于3层的IP地址数据转发的路由行为) 原始包交换 查两张表 在包交换过程中,数据包每经过一个路

    2024年02月22日
    浏览(47)
  • 数组(一篇带你掌握数组)

        在之前,我们想要存储一个数据的时候可以创建变量,例如存储一个整形的变量,我们使用int类型的变量来存储,那么如果存储一组相同类型的数据呢?这时我们就引入了 数组 的概念。 目录 一、一维数组的创建和初始化 1.1数组的创建 1.2 数组的初始化 1.3 一维数组的使

    2023年04月08日
    浏览(47)
  • Javaの一篇带你吃透接口

    随着接口的到来,JavaSE的学习笔记大结局也即将来临,最近的几篇博客写到了封装,继承,多态,抽象类等等,都循序渐进得介绍了这类的知识,大家如果接口这一块理解的很困难的话,建议去完善一下前面的知识哦 👉Java封装 👉静态成员 👉代码块 👉内部类 👉继承 👉多

    2023年04月08日
    浏览(62)
  • 【数据结构】一篇带你彻底了解栈

    栈:一种线性数据结构,其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶 (Top), 另一端称为栈底 [Bottom]。栈中的数据元素遵守后进先出LIFO(Last In First Out)的原则。即最后进入的元素最先被访问。 压栈:栈的插入操作叫做进栈/压栈

    2024年02月05日
    浏览(87)
  • 【数据结构】一篇带你彻底吃透 顺序表

    顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成数据的增删查改等功能。 顺序表一般可以分为: 静态顺序表:使用定长数组存储元素。 动态顺序表:使用动态开辟的数组存储。 而现实的顺序表大多数采用动态

    2023年04月19日
    浏览(79)
  • 【Python】一篇带你掌握数据容器之列表

    目录 前言: 一、列表 1.列表的定义 2.列表的下标索引 3.列表的常用操作 (1)index方法:查找某元素的下标 (2)修改特定位置下标的元素 (3)insert(下标,元素)方法:插入元素 (4)append(元素)方法:追加元素1 (5)extend(其他数据容器)方法:追加元素2 (6)del(列表

    2024年02月05日
    浏览(51)
  • [Linux 基础] 一篇带你了解linux权限问题

    Linux下有两种用户:超级用户(root)、普通用户。 超级用户:可以再linux系统下做任何事情,不受限制 普通用户:在linux下做有限的事情。 超级用户的命令提示符是“#”,普通用户的命令提示符是“ $ ” 命令: su [用户名] 功能: 切换用户。 例如,要从root用户切换到普通用

    2024年02月08日
    浏览(46)
  • Linux - 一篇带你读懂 Curl Proxy 代理模式

    curl 是一个很有名的处理网络请求的 类Unix 工具。出于某种原因,我们进行网络请求,需要设置代理。本文讲全面介绍如何为 curl 设置代理 设置代理参数 基本用法 设置 HTTP 代理 下面两种设置代理的方式是可以的 由于代理地址的默认协议为  HTTP, 所以可以省略,按照下面的

    2024年02月05日
    浏览(68)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包