C++中memcpy和memmove的區(qū)別總結(jié)
變態(tài)的命名
我們在寫程序時,一般講究見到變量的命名,就能讓別人基本知道該變量的含義。memcpy內(nèi)存拷貝,沒有問題;memmove,內(nèi)存移動?錯,如果這樣理解的話,那么這篇文章你就必須要好好看看了,memmove還是內(nèi)存拷貝。那么既然memcpy和memmove二者都是內(nèi)存拷貝,那二者究竟有什么區(qū)別呢?
先說memcpy
你有沒有好好的參加過一場C++筆試。讓你寫出memcpy的實現(xiàn),這是多么常見的筆試題啊?,F(xiàn)在,拿起你的演算紙和筆;是的,是筆和紙,不是讓你在你的IDE上寫。寫不出來?看下面吧:
void *mymemcpy(void *dest, const void *src, size_t count)
{
assert(dest != NULL || src != NULL);
char *tmp = (char *)dest;
char *p = (char *)src;
while (count--)
{
*tmp++ = *p++;
}
return dest;
}
memcpy的實現(xiàn)很簡單,一般在筆試時,出現(xiàn)寫源碼的題目,無非就是需要注意以下幾點:
1.確定函數(shù)原型;
2.判斷參數(shù)合法性;
3.邏輯實現(xiàn)(考慮各種情況,統(tǒng)稱邏輯實現(xiàn));
4.錯誤處理。
當然了,我的這個沒有錯誤處理,也不需要錯誤處理。上面,我寫出了memcpy的實現(xiàn)源碼,實現(xiàn)原理如下圖所示:

這樣下去,上面的代碼會運行的很好,如果出現(xiàn)下面的情況呢?

i、n、k的內(nèi)存和J、e、l的內(nèi)存地址重合了,現(xiàn)在再使用上面的代碼進行copy時,會出現(xiàn)什么問題呢?你有沒有想過這個問題。如果沒有,那就現(xiàn)在想想,不急著閱讀下面的內(nèi)容。
然后,我再留一個問題,上面的代碼中,為什么都需要將void *轉(zhuǎn)換成char *呢?比如:
char *tmp = (char *)dest;
可以留言回答哦。
再說memmove
memmove也是用來實現(xiàn)內(nèi)存的直接拷貝的。說起這個命名,我個人覺的多少還是有點坑的。既然memmove也是用來內(nèi)存數(shù)據(jù)移動的,那就先來看看memmove的實現(xiàn)源碼。
void *mymemmove(void *dest, const void *src, size_t count)
{
assert(dest != NULL || src != NULL)
if (dst < src)
{
char *p = (char *)dest;
char *q = (char *)src;
while (count--)
{
*p++ = *q++;
}
}
else
{
char *p = (char *)dest + count;
char *q = (char *)src + count;
while (count--)
{
*--p = *--q;
}
}
return dest;
}
從源碼看,memmove的確比memcpy復(fù)雜一些;再仔細一看,多了些什么?哦,多了一個else分支,而正是這個else分支,就處理了當src和dest的內(nèi)存重合的問題。
memcpy和memmove的比較
從實現(xiàn)源碼中的確能看出一些貓膩,當出現(xiàn)了src和dest的內(nèi)存有重合的時機時,memmove的處理規(guī)則是從后往前進行copy。當然了,重合的問題,需要考慮的以下兩種場合。

如圖所示,當出現(xiàn)(1)對應(yīng)的情況時,就需要先從src的頭部開始復(fù)制;也就是memmove源碼中的if分支,這部分源碼和memcpy的實現(xiàn)是一致的;當出現(xiàn)(2)對應(yīng)的情況時,就需要先從src的尾部開始復(fù)制,防止出現(xiàn)了覆蓋現(xiàn)象。這就是memmove比memcpy多的一個考慮點,所以說,在實際使用時,使用memmove是比memcpy更安全的。
總結(jié)
總結(jié)到了這里,我覺的我已經(jīng)把問題說清楚了。你說呢?如果你還有什么好的想法,歡迎你和我分享。
相關(guān)文章
C語言中的函數(shù)指針基礎(chǔ)學(xué)習(xí)教程
這篇文章主要介紹了C語言中的函數(shù)指針基礎(chǔ)學(xué)習(xí)教程,包括函數(shù)指針作為參數(shù)來傳遞等重要知識,需要的朋友可以參考下2016-04-04
C++ 中CloseHandle 函數(shù)--關(guān)閉一個句柄
這篇文章主要介紹了C++ 中CloseHandle 函數(shù)--關(guān)閉一個句柄的相關(guān)資料,需要的朋友可以參考下2017-05-05
Qt數(shù)據(jù)庫應(yīng)用之數(shù)據(jù)打印到pdf
因為xls打開以后用戶可以修改數(shù)據(jù)造假之類的,而pdf默認是不可編輯的,除非借助專業(yè)的工具,所以如果想要限定用戶導(dǎo)出數(shù)據(jù)不能被更改,那導(dǎo)出pdf是最佳選擇。所以本文將為代價介紹Qt實現(xiàn)數(shù)據(jù)打印到pdf的方法,需要的可以參考一下2022-01-01
Opencv學(xué)習(xí)教程之漫水填充算法實例詳解
這篇文章主要給大家介紹了Opencv學(xué)習(xí)教程之漫水填充算法的相關(guān)資料,文中給出了詳細的示例代碼供大家參考學(xué)習(xí),對大家具有一定的參考價值,需要的朋友們下面跟著小編一起來學(xué)習(xí)學(xué)習(xí)吧。2017-06-06

