c语言内存函数

目录

1.memcpy的使用和模拟实现

[1.1 代码演示](#1.1 代码演示)

[1.2 模拟实现](#1.2 模拟实现)

2.memmove的使用和模拟实现

[2.1 代码演示](#2.1 代码演示)

[2.2 模拟实现](#2.2 模拟实现)

3.memset的使用

[3.1 代码演示](#3.1 代码演示)

4.memcmp的使用

[4.1 代码演示](#4.1 代码演示)

1.memcpy的使用和模拟实现

复制代码
//函数原型
void* memcpy(void* destination,const char* source,size_t num);

功能:

memcpy是完成内存拷贝的,不关注内存中存放的数据是啥,一次性全拷贝

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

如果source和destination有任何重叠,复制的结果是未定义的。(内存重叠的情况使用memmove就行)。

memcpy的使用需要包含头文件<string.h>
参数:

destination:指针,指向目标空间,拷贝的数据存放在这里;

source: 指针,指向源空间,要拷贝的数据来自这里;

num: 要拷贝的数据所占据的字节数。

返回值:拷贝完成后,返回目标空间的起始地址。

1.1 代码演示
cpp 复制代码
#include<string.h>
int main()
{
	int dest[30] = { 0 };
	int src[] = { 1,2,3,4,5,6,7,8,9,10 };
	memcpy(dest, src, 20);
	for (int i = 0; i < 5; i++)
	{
		printf("%d ", dest[i]);
	}
	return 0;
}
1.2 模拟实现
cpp 复制代码
//模拟实现memcpy
#include<stdio.h>
#include<assert.h>
void* my_memcpy(void* dest, const void* str, size_t num)
{
	assert(str);
	void* ret = dest;
	while (num)
	{
		*(char*)dest = *(char*)str;
		dest = (char*)dest + 1;//强制类型转换是临时的
		str = (char*)str + 1;
		num--;
	}
	return ret;
}
int main()
{
	int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };
	int arr2[30] = { 0 };
	my_memcpy(arr2, arr1, 10 * sizeof(int));
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr2[i]);
	}
	return 0;
}

因为memcpy函数是按照字节进行拷贝的,所以我们需要将地址转成char*类型的,这样我们就可以完成任意类型数据的拷贝。

2.memmove的使用和模拟实现

cpp 复制代码
//函数原型
void* memmove(void* destination,const void* source,size_t num);

功能:

memmove函数也是完成内存块的拷贝的,和memcpy的差别是memmove函数处理的是原内存块和目标内存块是可以重叠的。

memmove函数的使用需要包含头文件<string.h>。
参数:

destination: 指针,指向目标空间,拷贝的数据存放到这里;

source: 指针,指向源空间 ,要拷贝的数据从这里来;

num:要拷贝数据占据的字节数。

返回值:

拷贝完成后,返回目标空间的起始地址。

2.1 代码演示
cpp 复制代码
#include<string.h>
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	memmove(arr, arr + 2, 20);
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}
2.2 模拟实现

代码一:

cpp 复制代码
//模拟实现memmove
#include<assert.h>
void* my_memmove(void* dest, const void* str, size_t num)
{
	assert(dest && str);
	void* ret = dest;
	if (dest<str || (char*)dest>(char*)str + num)
	{
		//前->后
		while (num--)
		{
			*(char*)dest = *(char*)str;
			dest = (char*)dest + 1;
			str = (char*)str + 1;
		}
		return ret;
	}
	else
	{
		//后->前
		while (num--)
		{
			*((char*)dest + num) = *((char*)str + num);
		}
		return ret;
	}
}
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	my_memmove(arr, arr + 2, 20);
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

代码二:

cpp 复制代码
//模拟实现memmove
#include<assert.h>
void* my_memmove(void* dest, const void* str, size_t num)
{
	assert(dest&&str);
	void* ret = dest;
	if (dest < str)
	{
		//前->后
		while (num--)
		{
			*(char*)dest = *(char*)str;
			dest = (char*)dest + 1;
			str = (char*)str + 1;
		}
		return ret;
	}
	else
	{
		//后->前
		while (num--)
		{
			*((char*)dest + num) = *((char*)str + num);
		}
		return ret;
	}
}
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	my_memmove(arr, arr + 2, 20);
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

3.memset的使用

cpp 复制代码
//函数原型
void* memset(void* ptr,int value,size_t num);

功能:
memset函数是用来设置内存块的内容的,将内存中指定长度的空间设置为特定的内容,是以字节为单位来设置的。

memset函数的使用需要包含头文件<string.h>。
参数:

ptr:指针,指向要设置的空间内存,也就是存放了要设置的内存空间的起始地址;

value:要设置的值,函数会把value值转换成unsigned char的数据进行设置, 也就是以字节为单位来设置内存块的

num:要设置的内存长度,单位字节。

返回值:返回要设置的内存空间的起始地址。

3.1 代码演示
cpp 复制代码
#include<string.h>
int main()
{
	char str[] = "helloworld";
	//现在我想将这个字符串全换成h
	memset(str, 'h', 10);
	printf("%s\n", str);
	return 0;
}

提醒:我们想将str[10]={1,2,3,4,5,6,7,8,9,10};这个数组中的数字全置为1,通过memset函数是不可能完成的,这是因为memset函数是按照字节来设置的,它会将每个字节都设置为1

4.memcmp的使用

cpp 复制代码
//函数原型
int memcmp(const void* ptr1,const void* ptr2,size_t num);

功能:

比较指定的两块内存块的内容,比较从ptr1和ptr2指针指向的位置开始,向后的num个字节。memcmp函数是一对字节一对字节往后比较。

memcmp需要包含头文件<string.h>。
参数:

ptr1:指针,指向一块待比较的内存块;

ptr2:指针,指向另一块待比较的内存块;

num:指定的比较长度,单位是字节。

返回值:

4.1 代码演示
cpp 复制代码
#include<string.h>
int main()
{
	char arr1[] = "hello world";
	char arr2[] = "hello bit";
	int ret=memcmp(arr2, arr1,5);
	if (ret > 0)
		printf(">");
	else if (ret < 0)
		printf("<");
	else
		printf("==");
	return 0;
}
相关推荐
蓝田生玉1234 小时前
BEVFormer论文阅读笔记
论文阅读·笔记
西瓜堆5 小时前
提示词工程学习笔记: 工程技术行业提示词推荐
笔记·学习
LawrenceLan5 小时前
Flutter 零基础入门(十一):空安全(Null Safety)基础
开发语言·flutter·dart
txinyu的博客5 小时前
解析业务层的key冲突问题
开发语言·c++·分布式
码不停蹄Zzz5 小时前
C语言第1章
c语言·开发语言
行者966 小时前
Flutter跨平台开发在OpenHarmony上的评分组件实现与优化
开发语言·flutter·harmonyos·鸿蒙
阿蒙Amon6 小时前
C#每日面试题-Array和ArrayList的区别
java·开发语言·c#
666HZ6666 小时前
数据结构2.0 线性表
c语言·数据结构·算法
SmartRadio6 小时前
ESP32添加修改蓝牙名称和获取蓝牙连接状态的AT命令-完整UART BLE服务功能后的完整`main.c`代码
c语言·开发语言·c++·esp32·ble
wxr06166 小时前
GOF笔记
笔记·适配器·ooad