亲宝软件园·资讯

展开

详解C语言中strcpy函数与memcpy函数的区别与实现

ZHENGZJM 人气:0

字符串拷贝函数(strcpy)

函数原型

由图可知,strcpy的形参是(目的地址,来源地址),返回值为char *。

应用实例

我们将一个存放“abcdefg”的字符数组arr的内容拷贝到字符数组brr之中。

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>
int main()
{
    char arr[10] = "abcdefg";
    char brr[10] = { 0 };
    strcpy(brr, arr);
    printf("%s", brr);
    return 0;
}

缺点

如果不是字符类型,还能这样拷贝吗?

我们发现,编译器直接报错了,因为形参跟实参不兼容。 strcpy具有局限性,只能适用于字符类型的拷贝。我们想要任意类型的都能实现拷贝,那怎么办呢?我们可以使用内存拷贝函数,不管三七二十一,你在这个内存单元上,我就把你拷贝过去,这样便实现了任意类型的拷贝 。

内存拷贝函数(memcpy)

函数原型

由图可知,memcpy的返回类型是void*,它的形参是(目标地址,需要复制的地址,字节大小)

应用实例

我们依然将一个存放“abcdefg”的字符数组arr的内容拷贝到字符数组brr之中。

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>
int main()
{
    char arr[10] = "abcdefg";
    int ret = strlen(arr);//求出字符串的长度
    char brr[10] = { 0 };
    memcpy(brr, arr,ret*sizeof(char));//字节数=个数*数据类型大小
    printf("%s", brr);
    return 0;
}

这次我们试试整型数组能不能拷贝过去。 

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>
int main()
{
    int arr[10] = {1,2,3,4,5,6,7,8,9,10};
    int brr[10] = { 0 };
    memcpy(brr, arr, 10*4);//字节数=个数*数据类型大小
    for (int i = 0; i < 10; i++)
    {
        printf("%d ", brr[i]);
    }
    return 0;
}

成功了,这说明memcpy的适用范围强于strcpy,那么memcpy是怎么实现的呢?

memcpy的模拟实现(my_memcpy)

实现样例

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

首先观察它形式

我们依葫芦画瓢,也写出类似的函数形式。 

void* my_memcpy(void* dest, void* src, size_t count)

具体实现

因为我们用的都是void*来接受参数,目的是保证兼容性(毕竟你不知道使用者使用的时候用的是什么数据类型),但计算机不知道void*指的是什么,所以需要强制类型转换。而强制类型转换,换成什么类型好呢?如果我们用int型,我们知道一个int元素占4个字节,那我们需要复制3个字节的东西咋办?所以,我们用只占一个字节的char型。

*(char*)dest = *(char*)src;

在拷贝完一个字节之后,指针向后移动一个字节

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

那么我们如何控制拷贝多少个字节呢?用while搭配count就行。

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

函数运行完了,我们随便返回一个值就行。

return 0;

加载全部内容

相关教程
猜你喜欢
用户评论