欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語言?模擬實(shí)現(xiàn)memcpy與memmove函數(shù)詳解

 更新時(shí)間:2022年04月08日 17:46:43   作者:不一樣的煙火a  
這篇文章主要介紹了C語言詳解如何模擬內(nèi)存函數(shù),用到了mencpy與memmove兩個(gè)函數(shù),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步

一、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)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++實(shí)現(xiàn)五子棋游戲

    C++實(shí)現(xiàn)五子棋游戲

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)五子棋游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • C++多線程實(shí)現(xiàn)綁定CPU的方法詳解

    C++多線程實(shí)現(xiàn)綁定CPU的方法詳解

    這篇文章主要為大家詳細(xì)介紹了C++如何利用多線程實(shí)現(xiàn)綁定CPU,文中的示例代碼簡潔易懂,具有一定的學(xué)習(xí)價(jià)值,感興趣的小伙伴可以跟隨小編一起了解一下
    2023-05-05
  • C/C++語言中全局變量重復(fù)定義問題的解決方法

    C/C++語言中全局變量重復(fù)定義問題的解決方法

    這篇文章主要給大家介紹了關(guān)于C/C++語言中全局變量重復(fù)定義問題的解決方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2018-01-01
  • C語言實(shí)現(xiàn)通訊錄系統(tǒng)程序

    C語言實(shí)現(xiàn)通訊錄系統(tǒng)程序

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)通訊錄系統(tǒng)程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-06-06
  • C++中的vector中erase用法實(shí)例代碼

    C++中的vector中erase用法實(shí)例代碼

    在vector數(shù)組中我們刪除數(shù)組經(jīng)常用的就是erase方法,但是earse的用法一不注意就會出錯(cuò),今天我就遇到了,所以在這里總結(jié)一下,避免大家用錯(cuò),對vector中erase用法感興趣的朋友跟隨小編一起看看吧
    2022-11-11
  • C++的頭文件和實(shí)現(xiàn)文件詳解

    C++的頭文件和實(shí)現(xiàn)文件詳解

    這篇文章主要介紹了C++的頭文件和實(shí)現(xiàn)文件詳解的相關(guān)資料,需要的朋友可以參考下
    2015-01-01
  • 用c 獲取文件MD5值的實(shí)現(xiàn)方法

    用c 獲取文件MD5值的實(shí)現(xiàn)方法

    本篇文章是對用c語言獲取文件MD5值的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • C++之值傳遞&指針傳遞&引用傳遞的示例詳解

    C++之值傳遞&指針傳遞&引用傳遞的示例詳解

    這篇文章主要為大家詳細(xì)介紹了C++中值傳遞、指針傳遞和引用傳遞的定義與使用,文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)C++有一定幫助,需要的可以參考一下
    2022-10-10
  • C語言格式輸出二進(jìn)制的2種方法總結(jié)

    C語言格式輸出二進(jìn)制的2種方法總結(jié)

    眾所周知C中以八進(jìn)制,十進(jìn)制和十六進(jìn)制都可以通過%o,%d和%x輕松實(shí)現(xiàn),然而唯獨(dú)沒有提供二進(jìn)制輸出的快速方式,下面這篇文章主要給大家介紹了關(guān)于C語言格式輸出二進(jìn)制的2種方法,需要的朋友可以參考下
    2022-08-08
  • C語言入門篇--字符串的基本理論及應(yīng)用

    C語言入門篇--字符串的基本理論及應(yīng)用

    本篇文章是c語言基礎(chǔ)篇,主要為大家介紹了C語言中字符串的基本理論及應(yīng)用,希望可以幫助大家快速入門c語言的世界,更好的理解c語言
    2021-08-08

最新評論