C語言?模擬實(shí)現(xiàn)memcpy與memmove函數(shù)詳解
一、memcpy函數(shù)的介紹
1.函數(shù)的聲明
void * memcpy ( void * destination, const void * source, size_t num );
2.函數(shù)功能與注意事項(xiàng)
- 函數(shù)memcpy從source的位置開始向后復(fù)制num個(gè)字節(jié)的數(shù)據(jù)到destination的內(nèi)存位置。
- 注意這個(gè)函數(shù)在遇到 '\0' 的時(shí)候并不會停下來。
- 如果source和destination有任何的重疊,復(fù)制的結(jié)果都是未定義的。
- memcpy函數(shù)可以拷貝任何的類型的數(shù)據(jù),不像strcpy函數(shù)只能拷貝字符串。
3.函數(shù)的使用
#include <stdio.h> #include <string.h>//使用memcpy函數(shù)時(shí)記得引用它的頭文件 int main() { int arr1[10] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; int arr2[5] = { 0 };//總共大小為20字節(jié) memcpy(arr1, arr2, 20//拷貝20個(gè)字節(jié)的數(shù)據(jù));//將arr2中的數(shù)據(jù)拷貝到arr1中 int i = 0; printf("拷貝后arr1中的數(shù)據(jù)為:"); for (i = 0; i < 10; i++) { printf("%d ", arr1[i]); } return 0; }
運(yùn)行結(jié)果:
二、模擬實(shí)現(xiàn)memcpy函數(shù)
1.模擬分析
1.因?yàn)槲覀儾恢牢覀円截惖氖鞘裁搭愋偷臄?shù)據(jù),可能是char類型的數(shù)據(jù),也可能是int類型的數(shù)據(jù),還有可能是double類型的數(shù)據(jù),這些不同類型數(shù)據(jù)的大小是不同的。為了實(shí)現(xiàn)一個(gè)能拷貝所有類型數(shù)據(jù)的memcpy函數(shù),我們就只能一個(gè)字節(jié)一個(gè)字節(jié)的拷貝,因?yàn)樽钚☆愋偷拇笮∈且粋€(gè)字節(jié),這樣就能將所有類型的數(shù)據(jù)都進(jìn)行拷貝了。
2.因?yàn)槲覀儾恢纻鞯絤emcpy函數(shù)的地址是什么類型,所以我們在接收傳過來的地址時(shí)要用void*類型的指針來接收。
3.由于我們只需要將源地址存儲的數(shù)據(jù)拷貝到目標(biāo)地址里面,所以只需要改變目標(biāo)地址處存儲的內(nèi)容,而不需要改變源地址處存儲的地址。所以我們就需要用const void*類型的指針來接收源地址。
4.為了實(shí)現(xiàn)鏈?zhǔn)皆L問,我們要將傳進(jìn)來的目標(biāo)起始地址(destination)返回。由于這個(gè)函數(shù)在執(zhí)行的時(shí)候會改變destination存儲的內(nèi)容,所以我們要重新創(chuàng)建一個(gè)void*類型的指針來存儲這個(gè)地址。
5.為了避免傳進(jìn)來的地址是空指針,我們需要用assert來斷言傳進(jìn)來的地址不是空指針。
2.模擬實(shí)現(xiàn)
#include<stdio.h> #include<assert.h> //模擬實(shí)現(xiàn)memcpy void* my_memcpy(void* dest, const void* scr, size_t count) { assert(dest && scr);//斷言傳進(jìn)來的地址不是空指針 void* ret = dest;//保存目標(biāo)起始地址 while (count--)//拷貝源地址存儲的數(shù)據(jù) { *(char*)dest = *(char*)scr; (char*)dest = (char*)dest + 1; (char*)scr = (char*)scr + 1; } return ret;//返回目標(biāo)起始地址 } //應(yīng)用模擬實(shí)現(xiàn)的函數(shù) int main() { int arr1[] = { 6, 6, 6, 6, 6, 6, 6, 6, 6, 6 }; int arr2[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; my_memcpy(arr2, arr1, 24);//拷貝6個(gè)字節(jié)的數(shù)據(jù) int i = 0; for (i = 0; i < 10; i++) { printf("%d ", arr2[i]); } return 0; }
運(yùn)行結(jié)果:
三、memmove函數(shù)的介紹
1.函數(shù)的聲明
void * memmove ( void * destination, const void * source, size_t num );
2.為什么會有memmove函數(shù)
為什么會有memmove這個(gè)函數(shù)呢,這個(gè)還要從上面的memcpy函數(shù)說起。因?yàn)閙emcpy函數(shù)不能將一個(gè)數(shù)組的中的數(shù)據(jù)拷貝到自身(也就是目標(biāo)數(shù)據(jù)是自己,源數(shù)據(jù)也是自己,只不過是一個(gè)數(shù)組里面不同的位置的數(shù)據(jù)拷貝到另外一個(gè)位置上),如果像這樣拷貝就會出現(xiàn)重疊拷貝,會導(dǎo)致結(jié)果不是我們預(yù)期的結(jié)果。
就像下面這個(gè)代碼:
//應(yīng)用模擬實(shí)現(xiàn)的memcpy函數(shù) int main() { int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; my_memcpy(arr + 2, arr, 24);//預(yù)期出現(xiàn)結(jié)果為1 2 1 2 3 4 5 6 9 10 int i = 0; for (i = 0; i < 10; i++) { printf("%d ", arr[i]);//實(shí)際出現(xiàn)結(jié)果 } return 0; }
運(yùn)行結(jié)果:
出現(xiàn)預(yù)期結(jié)果和實(shí)際結(jié)果不同的原因:
出現(xiàn)這種結(jié)果的原因就是因?yàn)閙emcpy函數(shù)將自身數(shù)據(jù)拷貝到自身不同位置的時(shí)候出現(xiàn)了重疊拷貝。源數(shù)據(jù)的起始地址為arr,目標(biāo)數(shù)據(jù)的起始地址arr + 2,當(dāng)我們一進(jìn)來memcpy這個(gè)函數(shù)的時(shí)候,我們就先將arr處的數(shù)據(jù)拷貝到arr + 2處,將arr + 1處的數(shù)據(jù)拷貝到arr + 3處,當(dāng)我們想要將arr + 2處的數(shù)據(jù)拷貝到arr + 4處的時(shí)候,我們發(fā)現(xiàn)arr + 2處的數(shù)據(jù)已經(jīng)被替換成了arr處的數(shù)據(jù)(1),于是我們就只能將1拷貝到arr + 4處;當(dāng)我們要將arr + 3處的數(shù)據(jù)拷貝到arr + 5處的時(shí)候,我們發(fā)現(xiàn)arr + 3處的數(shù)據(jù)早已被替換成了arr + 1處的數(shù)據(jù)(2),所以我們只能將2拷貝到arr + 5處,就像這樣反復(fù)的重疊拷貝,拷貝的數(shù)據(jù)一直都是1/2/1/2/1/2,直到拷貝完我們想要拷貝的字節(jié)數(shù)。
于是為了將自身的數(shù)據(jù)拷貝到自身不同的位置處,我們就需要用memmove函數(shù)來實(shí)現(xiàn),memmove函數(shù)就是為了解決上面這種問題而被創(chuàng)造的。
3.函數(shù)功能與注意事項(xiàng)
- memmove和memcpy的差別就是memmove函數(shù)處理的源內(nèi)存塊和目標(biāo)內(nèi)存塊是可以重疊的。
- 如果源空間和目標(biāo)空間出現(xiàn)重疊,就得使用memmove函數(shù)處理。
4.函數(shù)的使用
#include<stdio.h> #include<string.h>//使用memmove函數(shù)時(shí)記得引用它的頭文件 int main() { int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; memmove(arr + 2, arr, 24);//預(yù)期出現(xiàn)結(jié)果為1 2 1 2 3 4 5 6 9 10 int i = 0; for (i = 0; i < 10; i++) { printf("%d ", arr[i]);//實(shí)際出現(xiàn)結(jié)果 } return 0; }
這次我們發(fā)現(xiàn)用memmove函數(shù)來拷貝的預(yù)期結(jié)果和實(shí)際結(jié)果就一樣了,下面我們就講講memmove函數(shù)的模擬實(shí)現(xiàn)。
四、模擬實(shí)現(xiàn)memmove函數(shù)
1.模擬分析
1.將地址傳進(jìn)函數(shù)和函數(shù)接收地址的方法和上面的memcpy函數(shù)是一樣的,memcpy函數(shù)需要注意的地方memmove函數(shù)同樣需要注意,這里就不重復(fù)講了,嘿嘿。
2.memmove函數(shù)還需要注意的一點(diǎn)就是需要分析該怎么拷貝才不會重疊,下面為圖解:
情況一:dest小于等于src的地址
像下面這樣從前往后拷貝,這樣就不會重疊了。
情況二:dest大于scr的地址
像下面這樣從后往前拷貝,這樣就不會重疊了。
2.模擬實(shí)現(xiàn)
#include<stdio.h> #include<assert.h> //模擬實(shí)現(xiàn)memmove void* my_memmove(void* dest, const void* scr, size_t count) { assert(dest && scr);//斷言傳進(jìn)來的地址不是空指針 void* ret = dest; //保存目標(biāo)起始地址 if (dest <= scr)//從前往后拷貝 { while (count--) { *(char*)dest = *(char*)scr; (char*)dest = (char*)dest + 1; (char*)scr = (char*)scr + 1; } } else//從后往前拷貝 { while (count--) { *((char*)dest + count) = *((char*)scr + count); } } return ret; } //應(yīng)用模擬實(shí)現(xiàn)的函數(shù) int main() { int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; my_memmove(arr + 2, arr, 24);//預(yù)期出現(xiàn)結(jié)果為1 2 1 2 3 4 5 6 9 10 int i = 0; for (i = 0; i < 10; i++) { printf("%d ", arr[i]);//實(shí)際出現(xiàn)結(jié)果 } return 0; }
運(yùn)行結(jié)果
到此這篇關(guān)于C語言 模擬實(shí)現(xiàn)memcpy與memmove函數(shù)詳解的文章就介紹到這了,更多相關(guān)C語言 memcpy函數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!