C语言内存函数的使用及其模拟实现

 更新时间:2021年10月19日 10:32:40   作者:Ersansui  
这篇文章主要介绍了C语言内存函数的使用及其模拟实现,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

前言

在C语言中,我们除了会经常用到与字符相关的函数,我们还会使用到与内存相关的库函数。今天我们就来学习几个常见的内存函数吧!

在这里插入图片描述

memcpy

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

这是一个内存复制函数,该函数会从source的位置开始向后复制num个字节的数据到destination的内存位置。

这个函数在遇到 ‘\0' 的时候并不会停下来。

用法如下:

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[] = "Pierre de Fermat";
	char arr2[40];
	memcpy(arr2, arr1, strlen(arr1) + 1);
	printf("%s\n", arr2);
	return 0;
}

上述代码的arr2的输出结果为arr1中的内容。

我们要注意,在目标空间中,要保证有足够的空间放得下要拷贝的内容,以免造成越界。

模拟实现如下:

//memcpy的模拟实现
#include<stdio.h>
#include<string.h>
#include<assert.h>
//不会考虑内存重叠的情况
void* my_memcpy(void* dest, const void* source, size_t nums) {
	//先记录下dest的初始位置,便于后续返回
	void* ret = dest;
	//判断传入的两个指针是否为空
	assert(dest, source);
	//利用循环来控制拷贝字节的个数
	while (nums--) {
		//由于传入的是void*类型的指针,所以在使用前要强制转换为char*
		*(char*)dest = *(char*)source;
		dest = ((char*)dest) + 1;
		source = ((char*)source) + 1;
	}
	return ret;
}

memcmp

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

这个函数是用来比较指定字节数的内存空间是否相同。

返回值如下:

在这里插入图片描述

上图来自于这里

当ptr1的内容小于ptr2的内容,就返回一个小于零的数,当ptr1的内容大于ptr2的内容,就返回一个大于零的数,相等则返回零。

由于用法比较简单,就不进行演示啦!

模拟实现如下:

#include<stdio.h>
#include<string.h>
#include<assert.h>
int my_memcmp(const void* p1, const void* p2, size_t num) {
	//判断传入的两个指针是否为空
	assert(p1 && p2);
	//利用循环依次比较
	int i = 0;
	for (i = 0; i < num; i++) {
		if (*((char*)p1 + i) > (*(char*)p2 + i)) {
			return 1;
		}
		else if(*((char*)p1 + i) < (*(char*)p2 + i)){
			return -1;
		}
	}
	return 0;
}

memmove

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

上面我们已经学习过memcpy 函数的使用了,但是,如果我们的源空间和目标空间有重复的部分,那么memcpy这个函数就会出现错误,为了避免出错,我们就来学习一下memmove这个函数吧!

int main() {
	int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
	memmove(arr+2, arr, 16);
	printf("%s\n", arr);
	return 0;
}

上面的代码执行后,arr会变成{3,4,5,6,5,6,7,8,9,10}

这个函数的模拟实现要分情况讨论

1.如果dest在source前面,那么就有可能会出现有一部分内存重叠的情况,我们就需要在source中,先拷贝前面的数据在拷贝后面的数据,即从前往后拷贝。

2.如果dest在source后面,两个指针相减的数值小于要移动的字节数,那么也会有内存重叠的情况,那么此时,我们就需要先拷贝后面的数据,在拷贝前面的数据,即从后向前拷贝。

3.如果dest与source之间的差值大于要拷贝的字节数,那么此时就属于是两块不重叠的内存之间的拷贝,此时即使是用memcpy也不会有问题。

综上,我们可以已dest在source前后作为分界线。

如果dest在source前面,那么我们就从前往后拷贝,如果dest在source后面,那么我们就从后往前拷贝。

具体代码实现如下:

#include<stdio.h>
#include<string.h>
#include<assert.h>
void* my_memmove(void* dest, void* source, size_t num) {
	assert(dest &&  source);
	//记录初始dest的位置,方便后续返回
	void* ret = dest;
	//如果dest在source的前面,那么我们可以从前往后拷贝,防止数据被覆盖
	if (dest < source) {
		while (num--) {
			*(char*)dest = *(char*)source;
			dest = (char*)dest + 1;
			source = (char*)source + 1;
		}
	}
	else {
		while (num--) {
			//下面的num在第一次进入循环的时候已经减过1了
			*((char*)dest + num)= *((char*)source + num);
		}
	}
	return ret;
}

memset

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

这个函数一般用于初始化一段内存。

我们只需要传入内存空间的地址,需要初始化的字符,还有需要初始化的字节数即可。

由于这个函数的用法也比较简单,所以我们也不进行用法演示啦!

模拟实现如下:

#include<stdio.h>
#include<string.h>
#include<assert.h>
void* my_memset(void* dest, char ch, size_t num) {
	void* ret = dest;
	assert(dest);
	while (num--) {
		*(char*)dest = ch;
		((char*)dest)++;
	}
	return ret;
}

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!

相关文章

  • 深入解析C语言中常数的数据类型

    深入解析C语言中常数的数据类型

    C语言中常数的数据类型,需要的朋友可以过来参考下。希望对大家有所帮助
    2013-10-10
  • C++中的头文件与Extern(外部函数调用)方式

    C++中的头文件与Extern(外部函数调用)方式

    这篇文章主要介绍了C++中的头文件与Extern(外部函数调用)方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • C++超详细讲解RTTI和cast运算符的使用

    C++超详细讲解RTTI和cast运算符的使用

    RTTI(Runtime Type Identification)是“运行时类型识别”的意思。C++引入这个机制是为了让程序在运行时能根据基类的指针或引用来获得该指针或引用所指的对象的实际类型,cast强制转换运算符是一种特殊的运算符,它把一种数据类型转换为另一种数据类型
    2022-08-08
  • C++ GDI实现图片格式转换

    C++ GDI实现图片格式转换

    GDI+(Graphics Device Interface Plus)是一种用于图形绘制和图像处理的应用程序编程接口(API),在Windows平台上广泛使用,本文就来介绍一下如何使用GDI实现图片格式转换吧
    2023-12-12
  • C++索引越界的解决方法

    C++索引越界的解决方法

    本文主要介绍了C++索引越界的解决方法,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08
  • 链接库动态链接库详细介绍

    链接库动态链接库详细介绍

    静态链接库.lib和动态链接库.dll。其中动态链接库在被使用的时候,通常还提供一个.lib,称为引入库,它主要提供被Dll导出的函数和符号名称,使得链接的时候能够找到dll中对应的函数映射
    2012-11-11
  • C++中头文件与源文件的作用详解

    C++中头文件与源文件的作用详解

    这篇文章主要给大家介绍了关于C++中头文件与源文件的作用的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用C++具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-05-05
  • OpenCV实现多图像拼接成一张大图

    OpenCV实现多图像拼接成一张大图

    这篇文章主要为大家详细介绍了OpenCV实现多图像拼接成一张大图,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-01-01
  • C++ Boost Spirit进阶教程

    C++ Boost Spirit进阶教程

    Boost是为C++语言标准库提供扩展的一些C++程序库的总称。Boost库是一个可移植、提供源代码的C++库,作为标准库的后备,是C++标准化进程的开发引擎之一,是为C++语言标准库提供扩展的一些C++程序库的总称
    2022-11-11
  • C++类型转换运算符详解

    C++类型转换运算符详解

    这篇文章主要介绍了C++类型转换运算符的相关资料,希望通过本文大家能够掌握这部分内容,需要的朋友可以参考下,希望能够给你带来帮助
    2021-10-10

最新评论