Border.Effect绑定泄漏内存但Border.Background不

时间:2015-07-30 22:16:00

标签: c# wpf

使用更新的.NET 4.0,我发现了一个奇怪的内存泄漏,可以通过以下示例代码重现。

  • app.xml有一些应用程序范围的资源,它们绑定到app.xml.cs中的属性。创建泄漏的资源是<DropShadowEffect>,其Color依赖项属性绑定到App对象中的属性。
  • 主窗口有一个按钮,用于启动LeakWindow,其中使用app.xml中定义的资源。
  • 当泄漏窗口关闭时,它不会被垃圾收集。 :此 只有在泄漏窗口使用上述DropShadowEffect资源时才会发生。在SolidColorBrush Color也绑定到同一来源的情况下不会发生。

真的很感激,如果有人能告诉我为什么会在DropShadowEffect而不是SolidColorBrush上发生这种泄漏?

app.xml的

<Application x:Class="WpfSimple.App"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
    <Application.Resources>
        <!--this one make GC unable to collect LeakWindow-->
        <DropShadowEffect x:Key="AppDropShadowColor" 
              Color="{Binding Source={x:Static Application.Current}, Path=DropShadowColor, Mode=OneWay}" />
        <!--this one does not leak-->
        <SolidColorBrush x:Key="AppBackground"
              Color="{Binding Source={x:Static Application.Current}, Path=DropShadowColor, Mode=OneWay}" />
    </Application.Resources>
</Application>

App.xml.cs 启动MainWindow并为属性INotifyPropertyChanged实现DropShadowColor

 public partial class App : Application, INotifyPropertyChanged
    {
        protected override void OnStartup(StartupEventArgs e)
        {
            base.OnStartup(e);        
            // start main window
            var mainWindow = new MainWindow();
            mainWindow.Show();
        }

        private Color _dropShadowColor = Colors.Blue;    
        public Color DropShadowColor
        {
            get { return _dropShadowColor; }    
            set {
                _dropShadowColor = value;
                OnPropertyChanged("DropShadowColor");
            }
        }

        public event PropertyChangedEventHandler PropertyChanged;
        protected virtual void OnPropertyChanged(string propertyName)
        {
            PropertyChangedEventHandler handler = PropertyChanged;
            if (handler != null) { handler(this, new PropertyChangedEventArgs(propertyName)); }
        }
    }

MainWindow.xml和MainWindow.xml.cs 有一个用于创建LeakWindow的按钮,如下所示。

var win = new LeakWindow {Owner = this};
win.Show();

还有另一个按钮可以GC.Collect();

LeakWindow.xml

<Window x:Class="WpfSimple.LeakWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="Leak" Height="300" Width="300">
  <Grid>
    <!--leak-->
    <Border Width="200" Height="200" BorderThickness="1" BorderBrush="Black" Effect="{StaticResource AppDropShadowColor}"/>

    <!--no leak if comment out above and uncomment below-->
    <!--<Border  Width="200" Height="200" BorderThickness="1" BorderBrush="Black" Background="{StaticResource AppBackground}"/>-->
  </Grid>
</Window>

LeakWindow.xml.cs

 public partial class LeakWindow : Window
    {
        public LeakWindow()
        {
            InitializeComponent();
        }    

        ~LeakWindow()
        {
            Debug.WriteLine("LeakWindow finalized");
        }
    }

更新

  • 我的搜索显示这可能与此有关 DynamicResource\StaticResource cause memory leaks。但是在.NET 3.5(kb967328)的早期修补了那个。还尝试了该线程中提到的WalkDictionary方法但没有帮助。
  • 将绑定模式更改为OneTime也无济于事。
  • 切换到.NET 4.5(也修补到目前为止)并使用DynamicResource没有帮助。

进一步调查显示,泄漏是由EventHandler引用DropShadowEffect引起的。Border.Effect。可能是因DropShadowEffect中的绑定而导致的更改通知 仍然,奇怪的是为什么这只出现在Border.Effect而不是Border.Background上?

Workground
在app.xml中将x:Shared=false添加到<DropShadowEffect>可以解决此问题。我现在可以拥有应用程序范围内定义的资源,但会失去内存效率。

2 个答案:

答案 0 :(得分:1)

我认为问题是由DropShadowEffect附加到可视树的方式引起的。将DropShadowEffect移动到您的控件模板而不是将其作为资源可能会解决泄漏,但是您将失去该共享资源...

答案 1 :(得分:0)

        <Controls:MetroWindow x:Class="WpfApplication.Window"
                xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
                xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
                xmlns:Controls="clr-namespace:MahApps.Metro.Controls;assembly=MahApps.Metro" 
    ShowMaxRestoreButton="False" 
        ResizeMode="NoResize" ShowMinButton="False" 
        EnableDWMDropShadow="True" WindowStartupLocation="CenterScreen" Name="Window" 
Height="400" Width="567">

你可以尝试导入它有投影窗口的mah应用程序 一行EnableDWMDropShadow =“True”