在一次面试中突然被问到这个问题,当时还真不明白,回来在网上找到了说法:

 

因为malloc/free,new/delete都是调用HeapAlloc/HeapFree来实现来实现内存分配是释放的。

查看Windows的API可以看到,这两个函数都需要一个Heap的HANDLE做为参数。CRT库采用了全局变量来保存这个HANDLE。如果是CRT静态链接,CRT库的代码会链接到各个DLL中去,也包括这个全局变量。

也就是说,每个使用CRT静态链接的dll中都有一个自己的全局堆句柄,他们自己都在这个句柄上使用内存。当释放dll中分配的内存时由于使用的堆句柄不一致于是出错。

当使用CRT动态链接时,有于每个dll都是去调用CRT库的dll函数来分配和释放内存的,使用的是同一个句柄,所以就没有这个问题。

一般是哪里申请哪里释放,谁申请的内存由谁释放,这是封装的基本原则。   
  dll内部(对外隐藏)的内存分配,显然要dll自己处理啦。当然也可以由调用它的程序释放, 但不推荐,高内聚,且不安全 

   
  dll外部的由外部程序分配后传进dll,使用完后,可以由外部程序释放,也可以调用dll内部函数帮助释放。

  如果导出的是类,则可以在类中定义自我是释放的Release导出函数:   
  void   release()   
  {   
        delete   this;   
  }  

COM的内存释放就是采用这样的方法 

 

 

总结:静态链接的dll中申请的内存,必须由内部释放;动态链接的dll中申请的内存,可以由外部释放。

posted on 2015-06-19 21:19  奶味洋葱头  阅读(239)  评论(0编辑  收藏  举报