将所有标准事件处理程序替换为 WeakEventManager 或其变体是否安全?

Is it safe to replace all standard event handler to WeakEventManager or its variants?(将所有标准事件处理程序替换为 WeakEventManager 或其变体是否安全?)
本文介绍了将所有标准事件处理程序替换为 WeakEventManager 或其变体是否安全?的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

标准事件处理程序(使用运算符 +=)是 内存泄漏 原因之一(如果它没有取消注册/处置(使用 -= 运算符)).

Standard event handler (with operator +=) is one of the memory leakage cause (if it is not unregistered/disposed (with -= operator)).

微软通过 WeakEventManager 及其继承方式解决了这个问题,例如:PropertyChangedEventManager、CollectionChangedEventManager、CurrentChangedEventManager、ErrorsChangedEventManager 等等.

And Microsoft solved it with WeakEventManager and its inheritance like: PropertyChangedEventManager, CollectionChangedEventManager, CurrentChangedEventManager, ErrorsChangedEventManager and so on.

内存泄漏的简单示例代码为:

public class EventCaller
{
    public static event EventHandler MyEvent;
    
    public static void Call()
    {
        var handler = MyEvent;
        if (handler != null)
        {
            handler(null, EventArgs.Empty);
            Debug.WriteLine("=============");
        }
    }
}

public class A
{
    string myText;

    public A(string text)
    {
        myText = text;
        EventCaller.MyEvent += OnCall;

        // Use code below and comment out code above to avoid memory leakage.
        // System.Windows.WeakEventManager<EventCaller, EventArgs>.AddHandler(null, "MyEvent", OnCall);  
    }
    
    void OnCall(object sender, EventArgs e)
    {
        Debug.WriteLine(myText);
    }
    
    ~A()
    {
        Debug.WriteLine(myText + " destructor");
    }
}

void Main()
{
    var a = new A("A");
    var b = new A("B");
    EventCaller.Call();
    a = null;
    GC.Collect();
    EventCaller.Call();
}

输出为:

A
B
+++++++
A
B
+++++++

我们可以看到析构函数不会被调用.但是如果我们改变(通过注释未使用的代码):

We can see that the destructor will not be called. But if we change (by commenting the unused code) from:

    EventCaller.MyEvent += OnCall;

    System.Windows.WeakEventManager<EventCaller, EventArgs>.AddHandler(null, "MyEvent", OnCall);  

输出为:

A
B
+++++++
B
+++++++
A destructor
B destructor

在 A 为空后,它的事件处理程序将不再被调用.没有 -= 操作符的 A 和 B 将在不再使用后被丢弃.

After A is nulled then its event handler will not be called anymore. A and B will be disposed after not be used anymore without -= operator.

  1. 我是否可以安全地将所有 += 运算符替换为 System.Windows.WeakEventManager 以避免由于可能丢失事件注销和保存代码而导致的内存泄漏不应实现 IDisposable?

  1. Can I safely replace all += operator with System.Windows.WeakEventManager to avoid memory leakage due to probably missing event unregistration and saving code by should not implement IDisposable?

如果真的不安全,我应该考虑或注意什么?

If it is not really safe, what should I consider or notice?

推荐答案

我可以安全地将所有 += 运算符替换为 System.Windows.WeakEventManager 以避免由于可能丢失事件注销和保存代码而导致的内存泄漏,不应该实现 IDisposable?

Can I safely replace all += operator with System.Windows.WeakEventManager to avoid memory leakage due to probably missing event unregistration and saving code by should not implement IDisposable?

你可以吗?大概.你应该?可能不是.如果您确实有对事件处理程序的强引用,那么如果事件的发布者比事件的订阅者的寿命更长,那么您应该更喜欢取消订阅它,而不是用弱事件替换强引用.使用弱事件有副作用.其中之一是性能.另一个是语义差异.关于为什么.NET Framework中的事件实现默认不使用弱事件模式,你可能想参考以下问答:

Can you? Probably. Should you? Probably not. If you do have a strong reference to an event handler you should prefer unsubscribe from it if the publisher of the event lives longer than the subscriber of the event rather than replacing the strong reference with a weak event. There are side effects of using weak events. One of them is performance. Another is the semantic difference. You may want to refer to the following question and answers about why the implementation of events in the .NET Framework does not use the weak event pattern by default:

为什么C#中事件的实现默认不使用弱事件模式?

在某些情况下,您当然应该使用弱事件模式.一种这样的场景是 WPF 中的数据绑定,其中源对象完全独立于侦听器对象.但这并不意味着您应该始终使用弱事件模式.这也不意味着您应该停止关心在您的应用程序中处理订阅.

There are certainly certain scenarios where you should use the weak event pattern. One such scenario is the data binding in WPF where a source object is completely independent of the listener object. But this doesn't mean that you should always use the weak event pattern. And it also doesn't mean that you should stop caring about disposing subscriptions in your applications.

这篇关于将所有标准事件处理程序替换为 WeakEventManager 或其变体是否安全?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!

本站部分内容来源互联网,如果有图片或者内容侵犯您的权益请联系我们删除!

相关文档推荐

DispatcherQueue null when trying to update Ui property in ViewModel(尝试更新ViewModel中的Ui属性时DispatcherQueue为空)
Drawing over all windows on multiple monitors(在多个监视器上绘制所有窗口)
Programmatically show the desktop(以编程方式显示桌面)
c# Generic Setlt;Tgt; implementation to access objects by type(按类型访问对象的C#泛型集实现)
InvalidOperationException When using Context Injection in ASP.Net Core(在ASP.NET核心中使用上下文注入时发生InvalidOperationException)
LINQ many-to-many relationship, how to write a correct WHERE clause?(LINQ多对多关系,如何写一个正确的WHERE子句?)