【面接問題】memcpy関数の実装
面接でmemcpyの実現を聞かれたら、気をつけてください.ここに罠があります.
まず、標準memcpy()の説明を見てみましょう.
なお、この関数の動作は、アドレスが重なる場合には定義されていない.
実際にいう落とし穴もここにあり,自分でmemcpy()を実現する際にはアドレスが重なる場合を考慮する必要がある.
また、標準ライブラリでは、アドレスが重なるときのメモリコピー関数:memmove()も提供されていますが、memcpy()関数の書き換えを検討するのはなぜですか?
memmove()関数の実現効率の問題で,この関数はソース文字列を一時bufにコピーし,その後,一時bufから目的アドレスを書き,不要なオーバーヘッドを1回増加させた.
次にmemcpy()の実装を示し、標準ライブラリ関数と区別するために、そのラップ関数を実装します.
Memcpy()を使用した結果:
memcpy()を使用した結果:
標準ライブラリ関数のソース文字列がコピー中に汚染されていることがわかります.
2011-12-02任洪彩[email protected]
転載は出典を明記してください.
まず、標準memcpy()の説明を見てみましょう.
void *memcpy(void *dst, const void *src, size_t n);
//If copying takes place between objects that overlap, the behavior is undefined.
なお、この関数の動作は、アドレスが重なる場合には定義されていない.
実際にいう落とし穴もここにあり,自分でmemcpy()を実現する際にはアドレスが重なる場合を考慮する必要がある.
また、標準ライブラリでは、アドレスが重なるときのメモリコピー関数:memmove()も提供されていますが、memcpy()関数の書き換えを検討するのはなぜですか?
memmove()関数の実現効率の問題で,この関数はソース文字列を一時bufにコピーし,その後,一時bufから目的アドレスを書き,不要なオーバーヘッドを1回増加させた.
次にmemcpy()の実装を示し、標準ライブラリ関数と区別するために、そのラップ関数を実装します.
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
void *Memcpy(void *dst, const void *src, size_t size);
int main(int argc, char *argv[])
{
char buf[100] = "abcdefghijk";
//memcpy(buf+2, buf, 5);
Memcpy(buf+2, buf, 5);
printf("%s
", buf+2);
}
void *Memcpy(void *dst, const void *src, size_t size)
{
char *psrc;
char *pdst;
if(NULL == dst || NULL == src)
{
return NULL;
}
if((src < dst) && (char *)src + size > (char *)dst) //
{
psrc = (char *)src + size - 1;
pdst = (char *)dst + size - 1;
while(size--)
{
*pdst-- = *psrc--;
}
}
else
{
psrc = (char *)src;
pdst = (char *)dst;
while(size--)
{
*pdst++ = *psrc++;
}
}
return dst;
}
Memcpy()を使用した結果:
abcdehijk
memcpy()を使用した結果:
abadehijk
標準ライブラリ関数のソース文字列がコピー中に汚染されていることがわかります.
2011-12-02任洪彩[email protected]
転載は出典を明記してください.