gtk_object_sink的妙处
gtk_object_sink的妙处
转载时请注明出处和作者联系方式:http://blog.csdn.net/absurd
作者联系方式:Li XianJing <xianjimli at hotmail dot com>
更新时间:2006-1-15
在很长一段时间内,我一直不太明白gtk_object_sink这个函数的意义,它的实现也很简单:
void gtk_object_sink (GtkObject *object) { g_return_if_fail (GTK_IS_OBJECT (object)); if (GTK_OBJECT_FLOATING (object)) { GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING); g_object_unref (object); } } |
从代码的表面意义来看,也就是如果一个对象没有放入容器,就把它把反引用。功能很简单,但它的具体有什么用处呢。直到后来去阅读GTK+的代码,才明白它的妙处。
试想这样一种情况:把一个widget放入container中,调用者要不要反引用这个widget?
如果要,那也是合情合理的,container要保存widget的指针,当然要引用它(增加引用计数),在container销毁时反引用它。但是对调用者来说,就比较麻烦了,特别是widget比较多的情况,一个一个的去反引用它们,很容易遗漏,从而引起内存泄露。
如果不要,调用者是方便了,但container怎么办?保存widget的指针,但不增加的它的引用计数?而在container销毁时,却要反引用widget?虽然不是很合理,但在通常情况下也能工作。但如果把widget同时放入多个container中时,麻烦来了:这些container都没有引用widget,却在销毁时反引用它,程序一定会崩掉的。
怎么办呢?这时gtk_object_sink函数的妙处就显露出来了:对象在创建时,引用计数为1。在放入容器时,先增加对象的引用计数,然后调用gtk_object_sink。如果是第一次放入容器,OBJECT_FLOATING (object)为真,引用计数就减下来了,后面再加入容器时,OBJECT_FLOATING (object)为假,gtk_object_sink就没有任何效果。等效结果是:第一次放入容器不增加引用计数,以后放入容器都要增加引用计数。
这就是妙处?呵,其实只是sink这个术语有点生僻罢了。
~~end~~