问题描述:在windows窗体中加入一ActiveX控件,例如axCanlendar,显式释放控件的资源,清除对它的引用后,该控件仍然无法被GC所回收。
---------------------------------------------------------------------------------------------------------------------------------------------------------------------
问题重现:在Form中加入一个axCanlendar控件及一个Button。按钮单击的代码如下:
privatevoidbutton1_Click(objectsender,EventArgse)
{
this.Controls.Remove(this.axCalendar1);
WeakReferencewr=newWeakReference(this.axCalendar1,false);
axCalendar1.Dispose();
//Marshal.ReleaseComObject(axCalendar1.GetOcx());
axCalendar1=null;
inti=0;
while(wr.IsAlive)
{
i++;
GC.Collect();
MessageBox.Show(i.ToString());
}
}
--------------------------------------------------------------------------------------------------------------------------
问题分析:.net的GC是通过引用来判断某一资源是否可以回收的,对于非托管资源,我们一般需要显式调用其Dispose或close等函数来释放非托管资源。但是当我们显式调用axCanlendar的Dispose函数释放资源,并且将其设为null后,GC仍然无法将其回收,说明它依然被某一变量引用着。那么到底是什么引用着它呢。我们利用windbg可以轻松地查看它的根。
0:009> !dumpheap -type AxMSACAL.AxCalendar
可以看到axCalendar被LayoutEventArgs引用着。LayoutEventArgs是什么类型,而这个引用又是什么被添加的呢?带着这样的疑问,我们打开reflector,找到LayoutEventArgs类型。进行分析后发现,System.Windows.Forms.Control.PerformLayout函数会使用LayoutEventArgs类型。PerfomLayout共有三种形式:
Loading the heap objects into our cache.
Address MT Size
012c94f8 00978424 336 2 AxMSACAL.AxCalendar
0:009> !gcroot 012c94f8
ESP:12e5fc:Root: 012c8bd8(WindowsApplication4.Form1)->
012cbc78(System.Windows.Forms.LayoutEventArgs)->
012c94f8(AxMSACAL.AxCalendar)
PerformLayout(), PerformLayout(control, string), PerformLayout(LayoutEventArgs)
版权与免责声明
1、本站所发布的文章仅供技术交流参考,本站不主张将其做为决策的依据,浏览者可自愿选择采信与否,本站不对因采信这些信息所产生的任何问题负责。
2、本站部分文章来源于网络,其版权为原权利人所有。由于来源之故,有的文章未能获得作者姓名,署“未知”或“佚名”。对于这些文章,有知悉作者姓名的请告知本站,以便及时署名。如果作者要求删除,我们将予以删除。除此之外本站不再承担其它责任。
3、本站部分文章来源于本站原创,本站拥有所有权利。
4、如对本站发布的信息有异议,请联系我们,经本站确认后,将在三个工作日内做出修改或删除处理。
请参阅权责声明!