【内存函数】

目录

  1. memcpy使用和模拟实现
  2. memmove使用和模拟实现
  3. memset使用
  4. memcmp使用

1. memcpy使用和模拟实现

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

  • 函数memcpy从source的位置开始向后复制num个字节的数据到destination指向的内存的位置
  • 这个函数在遇到\0的时候并不会停下来
  • 如果source和destination有任何的重叠,复制的结果都是未定义的

1.1 使用

c 复制代码
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]);
}

拷贝arr1前5个数到arr2里

输出:

1.2 模拟实现

c 复制代码
void* my_memcpy(void * dest,const void * src,size_t num) {

	void* ret = dest;
	assert(dest);
	assert(src);

	while (num--) {
		*(char*)dest = *(char*)src;
		dest = (char*)dest + 1;
		src = (char*)src + 1;
	}

	return ret;
}

2. memmove使用和模拟实现

void * momove ( void* destination , cosnt void* source , size_t num) ;

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

2.1 使用

先看下面的代码

c 复制代码
int arr1[] = {1,2,3,4,5,6,7,8,9,10};
int arr2[10] = { 0 };
my_memcpy(arr1+3,arr1,20);

int i = 0;
for (i = 0; i < 10; i++) {
	printf("%d ", arr1[i]);
}

上面是想从arr1+3,也就是从4开始的位置,复制5个数据,也就是1,2,3,1,2,3,4,5,9,10。但最后的结果却不是这样的,原因就是当到重叠的位置时,数据已经被之前复制的覆盖过了,源数据已经被改变,所以出现错误

输出:

这时换成memove函数试试

c 复制代码
int arr1[] = {1,2,3,4,5,6,7,8,9,10};
int arr2[10] = { 0 };
memmove(arr1+3,arr1,20);

int i = 0;
for (i = 0; i < 10; i++) {
	printf("%d ", arr1[i]);
}

输出:

2.2 实现

对于一般的情况,我们直接拷贝数据就行,和memcpy一样。当数据有重叠时,可以分为三种情况

第一种,目标地址desc在源空间src的左边

这种情况下需要拷贝的数据在拷贝之前不会被覆盖,从前往后拷贝

第二种,目标地址desc在src+num中间,这种情况从前向后拷贝时原值会被替换,所以要从后往前拷贝

第三种,目标地址在src的右边,这个时候从左到右拷贝或从右到左都行

c 复制代码
void* my_memmove(void * dest,const void * src,size_t num) {

	assert(dest);
	assert(src);

	void* ret = dest;

	//从前往后拷
	if (dest < src || (char*)dest >= ((char*)src+num)) {

		while (num--) {
			*(char*)dest = *(char*)src;
			dest = (char*)dest + 1;
			src = (char*)src + 1;
		}
	}
	//从后往前拷
	else {

		dest = (char*)dest + num - 1;
		src = (char*)src + num - 1;

		while (num--) {
			*(char*)dest = *(char*)src;
			dest = (char*)dest - 1;
			src = (char*)src - 1;
		}
		
	}

	return ret;
}

3. memeset函数使用

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

memeset是用来设置内存的,将内存中的值以字节为单位设置为想要的内

c 复制代码
char str[] = "hello world";
memset(str,'x',6);
printf(str);

将字符串的前6个字符改为x

输出:

4. memcmp函数的使用

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

  • 比较从ptr1和ptr2开始,向后num个字节的内容
  • 返回值如下:
c 复制代码
char buffer1[] = "DWgaOtP12df0";
char buffer2[] = "DWGaOtP12df0";
int n;
n = memcmp(buffer1,buffer2,sizeof(buffer1));

printf("%d ", n);

上面的两个字符串,其中g小写和大写,由于小写的值更大,所以会返回大于0的数

输出:

相关推荐
我好喜欢你~17 分钟前
C#---StopWatch类
开发语言·c#
lifallen2 小时前
Java Stream sort算子实现:SortedOps
java·开发语言
IT毕设实战小研2 小时前
基于Spring Boot 4s店车辆管理系统 租车管理系统 停车位管理系统 智慧车辆管理系统
java·开发语言·spring boot·后端·spring·毕业设计·课程设计
cui__OaO3 小时前
Linux软件编程--线程
linux·开发语言·线程·互斥锁·死锁·信号量·嵌入式学习
鱼鱼说测试3 小时前
Jenkins+Python自动化持续集成详细教程
开发语言·servlet·php
艾莉丝努力练剑4 小时前
【洛谷刷题】用C语言和C++做一些入门题,练习洛谷IDE模式:分支机构(一)
c语言·开发语言·数据结构·c++·学习·算法
CHEN5_024 小时前
【Java基础面试题】Java基础概念
java·开发语言
Cx330❀5 小时前
【数据结构初阶】--排序(五):计数排序,排序算法复杂度对比和稳定性分析
c语言·数据结构·经验分享·笔记·算法·排序算法
杜子不疼.5 小时前
《Python学习之字典(一):基础操作与核心用法》
开发语言·python·学习
落霞的思绪6 小时前
Java设计模式详细解读
java·开发语言·设计模式