详解C语言中strcpy函数与memcpy函数的区别与实现
作者:ZHENGZJM
字符串拷贝函数(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;
到此这篇关于详解C语言中strcpy函数与memcpy函数的区别与实现的文章就介绍到这了,更多相关C语言 strcpy memcpy内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!