免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2705 | 回复: 1
打印 上一主题 下一主题

dlopen有可能导致内存泄漏 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2008-04-09 12:58 |只看该作者 |倒序浏览

下面我举一个常见的例子:

在动态库中,我要使用进程中唯一的对象mInstance,标准的写法:
liba.so的代码
static Myclass *mInstance;     //声明一个静态Myclass对象指针。

Myclass getInstance()
{
     if(NULL == mInstance)
     {
           mInstance = new Myclass;
      }
      return mInstance;
}

上面的代码,大家应该非常熟悉了,而在引入了dlopen之后,有可能会导致内存泄漏。

下面,我使用伪代码来说明这个问题。

1                         dlopen(liba.so);
2                         pInstance=getInstance();
……………….
3                         dlclose(liba.so);
………………….
4                         dlopen(liba.so);
5                         pInstance=getInstance();
………………….

我来解释一下有可能导致的内存泄漏。
1、               加载了动态库liba.so,同时初始化了其数据段,这时mInstance应该为空。
2、               在getInstance中,看到mInstance为空,则在堆中分配了一块内存,生成一个Myclass实例,同时为数据段的mInstance赋值。
3、               卸载了动态库liba.so,这时mInstance是不存在的,也就意味着我们丢失了在堆中生成的Myclass对象实例。
4、               加载了动态库liba.so,同时初始化了其数据段,这时mInstance应该为空。
5、               在getInstance中,看到mInstance为空,则在堆中又分配了一块内存,生成一个Myclass实例,同时为数据段的mInstance赋值。

这样,每做一次这样的时间循环,将会导致一个Myclass对象内存泄漏。

这个问题的实质是:
在我们的心目中,一个static的对象的生存周期是贯穿在进程始终的,实际上不是这样。在动态库中的static对象,其生命周期等于该动态库的生命周期。采用静态链接的方式,动态库的生命周期等于进程的声明周期;而采用动态加载的方式,则是不同的。

为了避免上面的问题出现,我们要在动态库卸载的时候,释放掉该块内存。

有两个方法:
1、  我们写一个释放内存的函数,在调用dlclose之前,调用该函数。
2、  我们在动态库的析构函数中,释放这块内存。
  void __attribute__ ((destructor)) my_fini(void);
在这里我推荐第二个方法,它对不需要上层用户做任何改动,对其完全透明。


本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u/30686/showart_523439.html

论坛徽章:
0
2 [报告]
发表于 2015-11-02 11:27 |只看该作者
也遇到了dlopen中的静态变量问题,Mark一下。
RTLD_NODELETE (since glibc 2.2)
Do not unload the shared object during dlclose().Consequently, the object's static variables are notreinitialized if the object is reloaded with dlopen() at alater time.
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP