Silverlight内存泄露(五)MEF等Ioc框架引起内存泄露-PartCreationPolicy

 

对象的创建可以使用new,也可以使用IOC架如:castle、MEF等,IOC创建的对象的生命周期,可能IOC负责管理,使用框架的开发者如果不弄清楚可能会造成内存泄露问题。

这些内存泄露问题并不是IOC框架的bug,只是开发者使用不当或者不注意造成的内存泄露问题。

以MEF为例说明我碰到的两种内存泄露问题。

内存泄露系列阅读提示:

一摸一样的对象图,有时候我们可以认为它是内存泄露,有时候又认为它不是内存泄露,这一切只是由于上下文不同,这一系列文章中ANTS Memoery Profle截图都是有特定上下文,单独看完全没有意义。如何确定是内存泄露?可以参考前面的文章。

对象以图的形式存在,Ants Memory Profile为了分析方便把这些图处理为树,让我们可以把注意力集中到分析的对象。但我们必须明白内存中对象关系构成图,也就是说ANTS的树状图只是内存中对象分布的一个局部,分析内存泄露时必须有全局观念,需要相关的几张图一起看,即使一张图也要整体看,这样才能分析内存泄露问题。

由于看一张图没什么意义,如果把多张图都贴出来,这文章就太难写了,即使多张图都贴出来,也不一定能表达清楚,分析内存泄露最重要的是经验。接下来的几篇会减少甚至不用这ANTS图。

PartCreationPolicy使用不当引起内存泄露

Shared:对象以Singleton方式创建。

Non Shared:每次请求时创建新对象。

Any Be default, CompositionContainer will use Shared, unless the ComposablePart or importer requests NonShared。默认值

在使用不同的组合方式时,PartCreationPolicy有不同的默认值,因此尽量不要使用默认值。

看下面的程序:

NavigationCommand:由于多个View页面都有需要导航,把导航都放到类NavigationCommand中,恰恰是这一做法引出了内存泄露问题。

 

 
 
[Export( typeof (NavigationCommand))]

public   class  NavigationCommand

{

private  RelayCommand < string >  readBookComamand  =   null ;

public  RelayCommand < string >  ReadBookComamand

{

get

{

if  (navigatToContentCommand  ==   null )

{

navigatToContentCommand 
=   new  RelayCommand < string >

(

=>

{

//
},

=>   ! string .IsNullOrWhiteSpace(p)

);

}

return  readBookComamand;

}

}

}
ViewModle

 

[ViewModelExport( typeof (BookPageSearcheViewModle),  " BookPageSearche " )]
[PartCreationPolicy(CreationPolicy.NonShared)]
public   class  BookPageSearcheViewModle: NavViewModel
{
public  RelayCommand < string >  ReadBookComamand
{
get
{
return  NavigationCommand. ReadBookComamand;
}
}
}
public   abstract   class  NavViewModel : ViewModelBase, IViewModel
{
///   <summary>
///  The navigation service allows you to navigate to an other ViewModel.
///   </summary>
[Import]
public  INavigationService NavigationService {  get set ; }
[Import]
public  NavigationCommand NavigationCommand
{
get ;
set ;
}
}

 

View

 

[ViewExport(ViewModelContract  =   typeof (BookPageSearcheViewModle))]
public   partial   class  BookPageSearcheView: Page, IView
{
public  BookPageSearcheView ()
{
InitializeComponent();
}
protected   override   void  OnNavigatedFrom(NavigationEventArgs e)
{
if  ( this .DataContext  is  ICleanup)
{
((ICleanup)
this .DataContext).Cleanup();
}
}
}
}

 

上例中NavigationCommand默认以Shared方式创建。

ANTS Memory Profiel 产生内存泄露的对象图:

5

可看出:BookPageSearcheViewModle 、Mef都引用了NavigationCommand,导致NavigationCommand没有被释放产生了内存泄露,在此只看MEF这一条路径。

我在处理这个内存泄露时太专注于BookPageSearcheViewModle对readBookCommand的引用了,而没有全面看这些图,断开BookPageSearcheViewModle对readBookCommand引用是最简单的处理方法,但是在处理其他ViewModel的时候会发现太多的Command需要处理,这时候才认识到断开引用的点错了。

因此一定要谨慎全面的看对象图,一个路径上包含n个点,断开任何点都能解决这个对象的泄露问题,但是一个程序有N个对象,这种只着眼当前对象的处理方式不能从根本上解决问题,必须找到内存泄露的根源

clip_image002

解决MEF这个泄露问题,最简单的是加上:[PartCreationPolicy(CreationPolicy.NonShared)]

还可以使NavViewModel类实现所有Command,把组合转换为继承,断开长引用。

结论

a) 不要盲目使用第三方框架。

b) 设计时组合优先于继承,但组合更容易产生内存泄露,因为任何一方如果是长时间存活,内存就不会被释放。继承则无这个问题。

c) 注意由IOC创建的对象生命周期,如果IOC创建的对象由容器管理生命期,可能需要调用IOC提供的相关方法执行对象的销毁。

e)MVVM 比Code behind 方式更容易产生内存问题。V、VM双向引用关系,更可能造成内存不被释放

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值