日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
C語言庫函數(shù)Memcpy和Memmove的區(qū)別,你知道多少?

 

 

基本概念闡述

memcpy和memmove都是 C 語言的庫函數(shù),相比于 strcpy和 strncpy只能針對于字符類型的數(shù)組(),這兩個函數(shù)可以拷貝其他類型的數(shù)組,對于 memcpy和 memmove的區(qū)別是什么呢?這里,在 Linux 里通過 man命令查看兩個函數(shù)的區(qū)別,查詢的結(jié)果如下所示,首先是 memcpy函數(shù)的闡述。

通過上述信息,可以知道,函數(shù)原型是:

 
 
 
 
  1. void *memcpy(void *dest, const void *src, size_t n); 

這個函數(shù)的功能如上面所說,就是復(fù)制src存儲區(qū)域 n個字節(jié)到dest區(qū)域,并且src和dest的內(nèi)存區(qū)域不能夠重疊。

緊接著來看memmove函數(shù),同樣的,來看Linux里的幫助手冊:

通過上述信息,可以知道,對于memmove的函數(shù)原型是:

 
 
 
 
  1. void *memmove(void *dest, const void *src, size_t n); 

具體函數(shù)是什么意思呢?通過上圖中的DESCRIPTION可以看到:

memmove() 函數(shù)將 n 個字節(jié)從內(nèi)存區(qū)域 src 復(fù)制到內(nèi)存區(qū)域 dest, 但是相比于memcpy函數(shù)不同的是,他的內(nèi)存區(qū)域可能會重疊:復(fù)制的過程就好比是將 src 中的字節(jié)首先被復(fù)制到一個不重疊的臨時數(shù)組中src 或 dest中,然后將字節(jié)從臨時數(shù)組復(fù)制到 dest。

實現(xiàn) memcpy 和 memmove及原理介紹

關(guān)于前面所敘述的內(nèi)存重疊的情況,會出現(xiàn)哪些問題呢?在論述這個問題之前,我們先來自己實現(xiàn) memcpy 和 memmove 函數(shù),當(dāng)然自己實現(xiàn)的大多數(shù)情況是沒有庫實現(xiàn)的那么嚴(yán)謹(jǐn)和完備的。首先是memcpy函數(shù)的實現(xiàn):

 
 
 
 
  1. void *memcpy(void *dest, const void *src, size_t count)   
  2. {   
  3.     if(dest == NULL || src == NULL || count <= 0)  return NULL;   
  4.     char *d = (char *)dest;   
  5.     char *s = (char *)src;   
  6.     while(count--)   
  7.     {   
  8.         *d++ = *s++;   
  9.     }   
  10.     return dest;   
  11. }   

代碼很容易理解,就不在這里進(jìn)行贅述了,其中,有一點也是筆者自己以前容易遺忘的一點,就是函數(shù)入口處對參數(shù)進(jìn)行檢查,不然會發(fā)生意想不到的錯誤。

接下來就是 memmove函數(shù)的實現(xiàn):

 
 
 
 
  1. void *memmove(void *dest, const void *src, size_t count)   
  2. {   
  3.     if(dest == NULL || src == NULL || count <= 0)  return NULL;   
  4.     if(dest < src)   
  5.     {   
  6.         char *d = (char *)dest;   
  7.         char *s = (char *)src;   
  8.         while (count--)   
  9.         {   
  10.             *d++ = *s++;   
  11.         }   
  12.     }   
  13.     else   
  14.     {   
  15.         char *d = (char *)dest + count;   
  16.         char *s = (char *)src + count;   
  17.         while (count--)   
  18.         {   
  19.             *--d = *--s;   
  20.         }   
  21.     }       
  22.     return dest;   
  23. }   

memmove 函數(shù)要相比于 memcpy函數(shù)的實現(xiàn)要復(fù)雜一點點:分成了目的地址在前還是在后兩種情況,如果是目的地址在前,那么就必須將src地址所在的字符串從前往后拷貝,反之,則必須將src所在的字符串從后往前拷貝。

如何解釋這一原因呢,我們從一個例子說起,下面是對應(yīng)的代碼:

 
 
 
 
  1. int main(int argc, char **argv)  
  2.     int arr[] = { 1,2,3,4,5,6,7,8,9,10 }; 
  3.     int i = 0; 
  4.     my_memcpy(arr + 2, arr, 20); 
  5.  
  6.     for (i = 0; i < 10; i++)  
  7.     { 
  8.         printf("%d ", arr[i]); 
  9.     } 
  10.  
  11.     return 0; 

可以看到代碼所實現(xiàn)的功能是,將arr數(shù)組中12345拷貝到 34567所在的地址中去,按照這樣一個思路,因該輸出的是:

 
 
 
 
  1. 1 2 1 2 3 4 5 8 9 10 

但是程序運行后輸出的是:

 
 
 
 
  1. 1 2 1 2 1 2 1 8 9 10 

這是為什么呢?筆者這里來圖解一下:

首先,將src地址的值賦值給dest,然后指針后移動,繼續(xù)下一次的賦值,此時數(shù)據(jù)就發(fā)生了變化,如下圖所示:

可以看到,此時 3 的位置變成了 1,繼續(xù)移動指針,也就有了如下的變化:

我們依據(jù)此原理,最后再移動三次指針,也就是如下所示的變化:

最終也就得到了上述的結(jié)果。

這種情況也就是dest在后,然后src在前的一種情況,如果是從前往后拷貝的話,也就會造成上述的問題,而解決的辦法就是從后往前拷貝,具體的過程,也如下圖所示:

可見,如果是此時 dest的地址在src的后面,那么就需要從后往前復(fù)制,這樣才不會導(dǎo)致數(shù)據(jù)覆蓋掉。

額外注意的一點,上文也提到了,就是說,對于 memmove 也不是一概而論的,如果是 dest的地址在前面,那么也還是需要從前往后復(fù)制才行。

至此,關(guān)于 memmove和 memcpy 的內(nèi)容就敘述完啦~

本文轉(zhuǎn)載自微信公眾號「wenzi嵌入式軟件」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系wenzi嵌入式軟件公眾號。


網(wǎng)站題目:C語言庫函數(shù)Memcpy和Memmove的區(qū)別,你知道多少?
網(wǎng)址分享:http://www.dlmjj.cn/article/dpgcjci.html