内存函数及模拟实现


前言

由于上篇文章中我们提到的 strcpy(),strcat(),strcmp()都只对字符串有效,那么库函数中是否存在另外一些函数更具通用性呢,一些既能对字符串,也能对浮点型,整型类型起作用的函数呢?答案是有的!它们就是内存函数,分别为memcpy(),memmove(),memset(),memcmp()

不允许内存重叠的内存拷贝函数:memcpy

在这里插入图片描述

将num字节的值从源指向的位置直接复制到目标指向的内存块。
该函数不检查源文件中的任何终止空字符,它总是精确地复制num个字节。
源文件和目标内存都至少应该大于等于num个字节,且两段内存不应该重叠!

使用案例

#include<stdio.h>
#include<string.h>
int main()
{
	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
	int arr2[] = { 5,4,3,2,1 };
	memcpy(arr1, arr2, sizeof(arr2));
	for (int i = 0; i < sizeof(arr1)/sizeof(int); i++)
	{
		printf("%d ", arr1[i]);
	}
	printf("\n");
	
	float arr3[10] = { 2.1,2.2,2.3,2.4,2.5 };
	float arr4[] = { 1.1,1.2,1.3,1.4,1.5,1.6 };
	memcpy(arr3, arr4, sizeof(arr4));
	for (int i = 0; i < sizeof(arr3)/sizeof(float); i++)
	{
		printf("%.1f ", arr3[i]);
	}
	printf("\n");
	
	char arr5[20] = "hello world";
	char arr6[] = "good night";
	char* ret = (char*)memcpy(arr5, arr6, 6);
	printf("%s\n", ret);

	return 0;
}

在这里插入图片描述

模拟实现

void* my_memcpy(void* dest, void* src, size_t num)
{
	assert(dest && src);
	void* ret = dest;
	while (num--)
	{
		*(char*)dest = *(char*)src;
		dest = (char*)dest + 1;
		src = (char*)src + 1;
	}
	return ret;
}

在这里插入图片描述

允许内存重叠的内存拷贝函数:memmove

在这里插入图片描述

memmove 的作用与memcpy的作用几乎相同,但是memmove允许存在内存重叠的情况

使用案例

int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	memmove(arr + 3, arr, 5 * sizeof(int));

	return 0;
}

在这里插入图片描述

模拟实现

思路:
对于内存的拷贝,有多种不同情况:
1.dest目标数组的首地址大于src拷贝数组的首地址,如果采用从前向后拷贝,则会造成src末尾数据被覆盖的现象。需要采用从后往前拷贝,以确保每个元素都能在不被前面数据覆盖下成功拷贝。
从前往后拷贝:1和2把4和5覆盖掉(不正常),变成1 2 3 1 2 6 7 8;再把3覆盖掉(正常),变成1 2 3 1 2 3 7 8;再把覆盖在4和5上的1 2覆盖在7和8上(不正常),变成1 2 3 1 2 3 1 2
在这里插入图片描述
2.dest目标数组的首地址小于src拷贝数组的首地址,如果采用从后往前拷贝,则会造成src的头部数据被覆盖
src的尾部数据7和8将src的头部数据4和5进行覆盖,导致在将src头部数据4和5拷贝进dest的头部时数据不同
在这里插入图片描述
3.没有内存重叠出现
无论上述哪种方法都能实现

void* my_memmove(void* dest, void* src, size_t num)
{
	assert(dest && src);
	void* ret = dest;
	//情况一
	if (dest > src)
	{
		while (num--)
		{
			*((char*)dest + num) = *((char*)src + num);
		}
	}
	//情况二和三
	else
	{
		while (num--)
		{
			*(char*)dest = *(char*)src;
			dest = (char*)dest + 1;
			src = (char*)src + 1;
		}
	}
	return ret;
}

内存设置函数:memset

在这里插入图片描述
作用:

将指针所指向的内存的num个字节设置为指定的值

使用案例
在这里插入图片描述
避坑题

现在我有一个具有20个int类型元素的数组,想要将它们全部设置为1,以下程序是否可以完成?

int main()
{
	int arr[20];
	memset(arr,1,sizeof(arr));
	return 0;
}

解析:

memset是一个内存设置函数,其每次设置的空间大小为1个字节,也就是说这个程序将把1按一个字节一个字节设置在内存中,每四个字节也就是一个int类型的元素设置后的数据为0000 0001 0000 0001 0000 0001 0000 0001,换算成16进制就是01 01 01 01,所以这个程序无法完成任务!
在这里插入图片描述

模拟实现

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

内存比较函数:memcmp

在这里插入图片描述

比较ptr1和ptr2所指向的num个字节,
如果ptr1大于ptr2,返回大于0的数
如果ptr1小于ptr2,返回小于0的数
如果ptr1等于ptr2,返回0
该函数与strcmp不同:遇到’\0’不会停止比较

使用案例
在这里插入图片描述

模拟实现

int my_memcmp(const void* ptr1, const void* ptr2, size_t num)
{
	assert(ptr1 && ptr2);
	size_t count = 1;
	while (count < num && *(char*)ptr1 == *(char*)ptr2)
	{
		count++;
		ptr1 = (char*)ptr1 + 1;
		ptr2 = (char*)ptr2 + 1;
	}
	return *(char*)ptr1 - *(char*)ptr2;
}

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值