• 内存重叠,直到做到一个笔试题才知道了什么是内存重叠。先上题目吧,是一个淘宝的笔试题,当时有点懵,不知道这个名词是啥子意思。
  • 题目:补充下面函数代码:
    如果两段内存重叠,用memcpy函数可能会导致行为未定义。 而memmove函数能够避免这种问题,下面是一种实现方式,请补充代码。

    • #include <iostream>
      using namespace std;
      void* memmove(void* str1,const void* str2,size_t n)
      {
      char* pStr1= (char*) str1;
      const char* pStr2=(const char*)str2;
      if ( ) {
      for(size_t i=;i!=n;++i){
      *(pStr1++)=*(pStr2++);
      }
      }
      else{
      pStr1+=n-;
      pStr2+=n-;
      for(size_t i=;i!=n;++i){
      *(pStr1--)=*(pStr2--);
      }
      }
      return ( );
      } 在上面的两个括号中插入对应的内容 答案:pstr1<pstr2 str1

      在这里我理解的内存重叠大致上应该是在strcpy以及memcpy等内存拷贝函数出现的问题。strcpy函数内存重叠可能会使程序崩溃,这里我先讲一个比较简单的例子来看一下。

    • #include <string.h>
      #include <stdlib.h>
      #include <stdio.h>
      int main(){
      char *p = NULL;
      p = (char *)malloc();
      memcpy(p,"",strlen(""));
      printf("before p = %s/n", p);
      strcpy(p+,p);//这重叠了
      printf("after p = %s/n", p);
      free(p);
      }

      上面的这个例子中发生了内存重叠问题,这就导致了拷贝发生了异常,不会拷贝一样的数据。我跑这个代码的时候居然没有报错,程序也没炸,我也就有点好奇了。按理来memcpy,strcpy这两个函数没有对内存重叠进行处理。使用这两个函数的时候只有程序员自己保证源地址与目标地址内存不重叠。所以当会发生内存重叠的时候最好使用memmov函数进行内存拷贝。

      自己查了一些资料,原来memcpy有一个长度参数,只拷贝cnt个字节就结束了,所以会得到正确的结果,但是strcpy函数知道拷贝到\0这个标志符才会结束,所以就会导致程序崩溃了。

  •   memcpy和memmov函数原型和区别
    •   1.memmove

      函数原型:void *memmove(void *dest, const void *source, size_t count)

      返回值说明:返回指向dest的void *指针

      参数说明:dest,source分别为目标串和源串的首地址。count为要移动的字符的个数

      函数说明:memmove用于从source拷贝count个字符到dest,如果目标区域和源区域有重叠的话,memmove能够保证源串在被覆盖之前将重叠区域的字节拷贝到目标区域中。

    •   2.memcpy

      函数原型:void *memcpy(void *dest, const void *source, size_t count);

      返回值说明:返回指向dest的void *指针

      函数说明:memcpy功能和memmove相同,但是memcpy中dest和source中的区域不能重叠,否则会出现未知结果。

  •   实现一个memmov函数
    •   

      #include <iostream>
      #include <string.h>
      using namespace std; void *memmove(void * dst, const void *src, size_t count){
      //特殊情况错误处理
      if (src == NULL || dst == NULL)
      return NULL;
      unsigned char *pdst = (unsigned char *)dst;
      const unsigned char *psrc = (const unsigned char *)src; //判断内存是否重叠
      bool flag1 = (pdst >= psrc && pdst < psrc + count);
      bool flag2 = (psrc >= pdst && psrc < pdst + count);
      //上面两个标志其中有一个成立,保证内存并不是重叠的,与拷贝的长度有关 if (flag1 || flag2){//内存重叠
      //倒序拷贝
      while (count){
      *(pdst + count - ) = *(psrc + count - );
      count--;
      }
      }
      else{//不重叠
      while (count--){
      *pdst = *psrc;
      pdst++;
      psrc++;
      }
      }
      return dst;
      } int main(){
      //内存重叠的情况
      char str[] = "hello world";
      memmove(str + , str, );
      cout<<"memmove result is :"<<str<<endl; //内存不重叠
      char str2[] = "hello world";
      char str3[] = "you are ";
      memmove(str2, str3, );
      cout<<"memmove result is :"<<str2<<endl;
      }

      实现思路就是要去判断是否内存重叠,重叠的话就倒序的拷贝,非重叠的话就从前往后拷贝就行了。这个例子也很好的解释了前面的那个题目的情况。其实前面的那个情况还可以用一个图片来解释。

    • 这张图片就是这两种情况的实现。

C++中的内存重叠问题的更多相关文章

  1. C++中两块内存重叠的string的copy方法

    如果两段内存重叠,用memcpy函数可能会导致行为未定义. 而memmove函数能够避免这种问题,下面是一种实现方式: #include <iostream> using namespac ...

  2. Windows内核中的内存管理

    内存管理的要点 内核内存是在虚拟地址空间的高2GB位置,且由所有进程所共享,进程进行切换时改变的只是进程的用户分区的内存 驱动程序就像一个特殊的DLL,这个DLL被加载到内核的地址空间中,Driver ...

  3. memmove 和 memcpy的区别以及处理内存重叠问题

    区别: memcpy和memmove()都是C语言中的库函数,在头文件string.h中,作用是拷贝一定长度的内存的内容,原型分别如下: void *memcpy(void *dst, const v ...

  4. [整理]内存重叠之memcpy、memmove

    函数原型: void *memcpy( void *dest, const void *src, size_t count ); void *memmove( void* dest, const vo ...

  5. strcpy,memcpy,memmove和内存重叠分析

    一:strcpy函数用法和实现: /* GNU-C中的实现(节选): */ char* strcpy(char *d, const char *s) { char *r=d; while((*d++= ...

  6. C语言标准库函数memcpy和memmove的区别以及内存重叠问题处理

    ①memcpy()和memmove()都是C语言中的标准库函数,定义在头文件string.h中,作用是拷贝一定长度的内存的内容,原型分别如下: void *memcpy(void *dst, cons ...

  7. 转:Linux中的内存管理

    前一段时间看了<深入理解Linux内核>对其中的内存管理部分花了不少时间,但是还是有很多问题不是很清楚,最近又花了一些时间复习了一下,在这里记录下自己的理解和对Linux中内存管理的一些看 ...

  8. C++ 浅析调试,内存重叠查看

    这里举个例子查看内存, 环境为:vs 2017 测试为strcpy[因为测试老api,需要在 预处理中 添加 _CRT_SECURE_NO_WARNINGS ] 测试问题:内存溢出 源码: #incl ...

  9. 在 CUDA C/C++ kernel中使用内存

    在 CUDA C/C++ kernel中使用内存 如何在主机和设备之间高效地移动数据.本文将讨论如何有效地从内核中访问设备存储器,特别是 全局内存 . 在 CUDA 设备上有几种内存,每种内存的作用域 ...

随机推荐

  1. .net remoting和wcf自托管——一个bug引发的警示

    一.解决问题,需要深入,并从细节入手,多从代码找原因,不能认为代码是死的,不会出错: 之前代码都运行良好,突然某一天,在我电脑上出问题了.出了问题,那就应该找出原因.其实这个问题,本身并不难,好歹给你 ...

  2. zookeeper环境及dubbo-admin管理平台搭建

    一. Zookeeper的安装使用 1.1 Zookeeper介绍 Zookeeper是一个分布式的,开放源码的分布式应用程序协调服务,是Google的Chubby一个开源的实现,是Hadoop和Hb ...

  3. CentOS6.5 下MySQL傻瓜式安装

    为了为服务器上装mysql我先在虚拟机上练习了一下特此记录并分享; 注:参考文章https://www.cnblogs.com/xiaoluo501395377/archive/2013/04/07/ ...

  4. .Net学习资源整理

    .Net学习资源整理 ASP.NET Core

  5. Powershell使用SSH

    在编写Powershell脚本时,有时会需要和Linux机器进行交互.所以这时就需要在Powershell中使用SSH. 本文将介绍如何在Powershell中安装SSH模块,以及如何使用ssh命令. ...

  6. sleep 和 usleep的实现方法

    一.sleep 和 usleep 1.不属于系统调用,是glibc 库函数实现的: 2.glibc函数库中通过调用内核的nanosleep实现的: 3.内核nanosleep通过调用 hrtimer_ ...

  7. 侯捷STL学习(二)--序列容器测试

    第六节:容器之分类和各种测试(四) stack不提供iterator操作,破坏了容器的独特性,先进先出. 使用容器multiset(允许元素重复) 内部是红黑树,insert操作就保证了排好了序. 标 ...

  8. Canvas 与 SVG 的比较

    Canvas:<canvas> 标签定义图形(只是图形容器),比如图表和其他图像,您必须使用脚本 (通常是JavaScript)来绘制图形.默认情况下 <canvas> 元素没 ...

  9. DVWA平台v1.9-Command Injection

    命令拼接: &:简单的拼接,第一条命令和第二条命令间没有什么制约关系 &&:第一条命令执行成功了,才会执行第二条命令 |:第一条命令的输出作为第二条命令的输入 ||:第一条命令 ...

  10. 分布式爬虫搭建系列 之一------python安装及以及虚拟环境的配置及scrapy依赖库的安装

    python及scrapy框架依赖库的安装步骤: 第一步,python的安装 在Windows上安装Python 首先,根据你的Windows版本(64位还是32位)从Python的官方网站下载Pyt ...