使用更新的.NET 4.0,我发现了一个奇怪的内存泄漏,可以通过以下示例代码重现。
<DropShadowEffect>
,其Color
依赖项属性绑定到App对象中的属性。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");
}
}
更新
OneTime
也无济于事。DynamicResource
没有帮助。进一步调查显示,泄漏是由EventHandler
引用DropShadowEffect
引起的。Border.Effect
。可能是因DropShadowEffect
中的绑定而导致的更改通知
仍然,奇怪的是为什么这只出现在Border.Effect
而不是Border.Background
上?
Workground
在app.xml中将x:Shared=false
添加到<DropShadowEffect>
可以解决此问题。我现在可以拥有应用程序范围内定义的资源,但会失去内存效率。
答案 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”