memmove 与memcpy

来源:互联网 发布:西安软件新城 2017 编辑:程序博客网 时间:2024/06/09 17:09

说下memmove ,简单解释下,如果目标地址dst在源地址src 到src +count 之间的话需要从后面向前拷贝 (不然dst的后部分得不到需要的结果,成了src的前部分了,dst前面可能出现传播效应)

dst = (char *)dst + count - 1;
src = (char *)src + count - 1;
然后

 while (count--) {
                        *(char *)dst = *(char *)src;
                        dst = (char *)dst - 1;
                        src = (char *)src - 1;
                }

这样目标是获得需要的结果,src后部分被修改。

否者就从地地址到高地址来拷贝,如

  while (count--) {
                        *(char *)dst = *(char *)src;
                        dst = (char *)dst + 1;
                        src = (char *)src + 1;
                }

这样目标是获得需要的结果,如果dst <src && dst + count >dst 的话,src前部分会被修改。


源码如下

void * __cdecl memmove (

        void * dst,
        const void * src,
        size_t count
        )
{
        void * ret = dst;

        if (dst <= src || (char *)dst >= ((char *)src + count)) {
                /*
                 * Non-Overlapping Buffers
                 * copy from lower addresses to higher addresses
                 */
                while (count--) {
                        *(char *)dst = *(char *)src;
                        dst = (char *)dst + 1;
                        src = (char *)src + 1;
                }
        }
        else {
                /*
                 * Overlapping Buffers
                 * copy from higher addresses to lower addresses
                 */
                dst = (char *)dst + count - 1;
                src = (char *)src + count - 1;


                while (count--) {
                        *(char *)dst = *(char *)src;
                        dst = (char *)dst - 1;
                        src = (char *)src - 1;
                }
        }

        return(ret);

}

对比下,memcpy 就没考虑那麽多,都是从低地址到高地址的拷贝所以必须不能出现

dst >=src && dst <src +count 的情况,否者dst就不是想要的结果。

void * __cdecl memcpy (
        void * dst,
        const void * src,
        size_t count
        )
{
        void * ret = dst;

        /*
         * copy from lower addresses to higher addresses
         */
        while (count--) {
                *(char *)dst = *(char *)src;
                dst = (char *)dst + 1;
                src = (char *)src + 1;
        }
        return(ret);
}


原创粉丝点击