您的位置:首页 > 其它

.Net内存优化的几点经验

2015-01-16 22:32 155 查看
  以前从来没有想过.Net开发居然存在内存无法释放的问题,总是认为GC给我处理好了一切。现在GIS二次开发结合三维球开发,没有想到存在如此严重的内存增长,很快内存就不够用了,导致系统各种不稳定。球体和三维模型就开始闪烁,出现无法创建D3D或GDI+设备,OutOfMemory等错误。最近一直为内存优化的事情头疼,虽然优化了部分内容,问题依然没有解决。

  还是总结了一下最近优化的经验:

1.慎重使用单例,单例会始终保持一个静态对象的引用,内存始终不释放,同时单例类的所有成员变量也不会释放。如单例窗体,解决方法是在Dispose方法中将静态引用置为null。

2.字符对象使用不当:过多太长字符串,占用大对象堆;同时尽量不使用字符串加,改用StringBuilder

3.流对象没有关闭。如文件流,网络传输流(HttpResponse,FtpResponse)

4.坑人的Select * from XX,应该避免使用这种代码,尤其是有Blob字段时候。

5.读取文件和流时避免new 过大的byte数组,应该分段读取。

6.事件的挂接。有两种情况:

A对象作为参数传入B对象,在B对象中为A挂接了事件,如果没有正确移除事件,会导致B对象内存无法回收。例如B为窗体变量,B被Dispose后,内存仍然不会回收。

A对象通过B的方法获取对象C,而C在B中挂接了事件,此时即使将B设为null,也无法将B对象的实例内存回收。

7.关于COM,目前项目中大量使用了COM组件,包括DEV、ArcEngine、Slimdx封装的Direct3D、Excel Interop、FlexCell.Net。

  感觉COM,总是内存不释放!不知道怎么回事,即使调用了Marshal.FinalReleaseComObject(),或者Marshal.ReleaseComObject(),感觉RCW的终结器始终没有被调用。ArcEngine在新建图层、删除图层、查询会有显著的内存增长,而且内存第一次明显增长,以后会缓慢增长,增长之后不会释放。

  造成的问题还包括启动读取xls文件的Excel.exe进程无法关闭、WorkspaceFactory占用影像tiff文件,提示文件被另外一个进程占用,无法删除。

  Excel进程关闭可以通过如下的方法解决,在使用完读取写入的方法后调用。

  对于GC.Collect()方法,在确定已经有大量对象被置空时,可以手工调用参考文章

  WorkspaceFactory占用影像tiff文件也可以通过释放RCW来解决,但是内存没有办法释放。

public void Quit()
{
try
{
_Application.Quit();
if (_Range != null)
{
Marshal.FinalReleaseComObject(_Range);
_Range = null;
}
if (_Worksheet != null)
{
Marshal.FinalReleaseComObject(_Worksheet);
_Worksheet = null;
}
if (_Workbook != null)
{
Marshal.FinalReleaseComObject(_Workbook);
_Workbook = null;
}
if (_Workbooks != null)
{
Marshal.FinalReleaseComObject(_Workbooks);
_Workbooks = null;
}
if (_Application != null)
{
Marshal.FinalReleaseComObject(_Application);
_Application = null;
}

}
catch { }
GC.Collect();
GC.WaitForPendingFinalizers();
}


希望大家给我点建议。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: