C/C++ string.h库中的memcpy()和memmove()

本文详细介绍了C语言中的memcpy和memmove函数,用于处理任意类型数组的复制,强调了它们在处理数组时的不同策略,特别是当数组可能存在重叠时的行为。通过示例代码展示了如何正确使用这两个函数以避免数据丢失。
摘要由CSDN通过智能技术生成

    不能把一个数组赋给另一个数组,所以要通过循环把数组中的每个元素赋给另一个数组相应的元素。有一个例外的情况是:使用strcpy()和strncpy()函数来处理字符数组。

    memcpy()和memmove()函数提供类似的方法处理任意类型的数组,下面是这两个寒素的原型:
        void *memcpy(void *restrict s1, const void *restrict s2, size_t n);
        void *memmove(void *s1, const void *s2, size_t n);

    这两个函数都从s2指向的位置拷贝n字节到s1指向的位置,而且都返回s1的值。所不同的是memcpy()的参数带关键字restrict,即memcpy()假设两个内存区域之间没有重叠。

    而memmove()不做这样的假设,所以拷贝过程类似于先把所有字节拷贝到一个临时缓冲区,然后再拷贝到最终目的地。

    如果使用memcpy()时,两区域出现重叠会怎样?其行为时未定义的。这意味着该函可能正常工作。也可能失败。编译器不会再本该不使用memcpy()时禁止你使用,作为程序员,再使用该函数时有责任确保两个区域不重叠。

    由于这两个函数设计用于处理任何数据类型,所有它们的参数都是两个指向void的针。

    C允许把任何类型的指针赋给void*类型的指针。如此宽容导致函数无法知道待拷贝据的类型。因此这两个函数使用第3个参数指明待拷贝的字节数。

    需要注意的时,对数组而言,字节数与元素的个数不同。如果要拷贝数组中10个double
类型的元素,要使用10*sizeof(double),而不是10.

示例源码:

// Len_memcpy.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//

#include <iostream>
#define SIZE 10
#include <string>
using namespace std;

void PrintD(string s, int* data, int len)
{	
	printf("%s\t", s.c_str());
	for (int i = 0; i < len; i++)
	{
		printf("%d\t", data[i]);
	}
	printf("\n");
}
void PrintD(string s, double* data, int len)
{
	printf("%s\t", s.c_str());
	for (int i = 0; i < len; i++)
	{
		printf("%.2lf\t", data[i]);
	}
	printf("\n");
}
int main()
{
	int nValues[SIZE] = { 1,2,3,4,5,6,7,8,9,10 };
	double dValues[SIZE] = { 1.1, 2.2, 3.3, 4.4, 5.5,6.6,7.7,8.8,9.9, 0.1 };
	
	int memcpyN[SIZE] = {0};          // 使用memcpy拷贝数据到这里
	double memcpyD[SIZE] = {0.0};
	
	int memmoveN[SIZE] = { 0 };		 // 使用memmove拷贝数据到这里
	double memmoveD[SIZE] = { 0.0 };

	printf(" copy before: \n");
	PrintD("Values  ", nValues, SIZE);
	PrintD("dValues ", dValues, SIZE);
	PrintD("targetN ", memcpyN, SIZE);
	PrintD("targetD ", memcpyD, SIZE);

	printf("\n after memcpy copy: \n");
	memcpy(memcpyN, nValues, sizeof(int)*SIZE);
	memcpy(memcpyD, dValues, sizeof(int)*SIZE); // double比size占位宽,这里会丢失数据
	PrintD("Values  ", nValues, SIZE);
	PrintD("dValues ", dValues, SIZE);
	PrintD("memcpyN ", memcpyN, SIZE);
	PrintD("memcpyD ", memcpyD, SIZE);

	printf("\n after memmove copy: \n");
	memcpy(memmoveN, nValues, sizeof(int)*SIZE);
	memcpy(memmoveD, dValues, sizeof(double)*SIZE); // 这里不会再丢失数据
	PrintD("Values  ", nValues, SIZE);
	PrintD("dValues ", dValues, SIZE);
	PrintD("memmoveN", memmoveN, SIZE);
	PrintD("memmoveD", memmoveD, SIZE);
}

执行结果:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

WendyWJGu

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值