首页 / C# / C#:如何避免这种潜在的内存泄漏
C#:如何避免这种潜在的内存泄漏
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了C#:如何避免这种潜在的内存泄漏,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含2321字,纯文字阅读大概需要4分钟。
内容图文
![C#:如何避免这种潜在的内存泄漏](/upload/InfoBanner/zyjiaocheng/969/a06a4f4f0d4748c98d01d249153ec165.jpg)
我们假设我有一个C#类,如下所示:
public class MyClass {
public SomeObject TheObject { get; }
public MyClass() {
TheObject = new SomeObject();
TheObject.MyEvent += MyEventHandler;
}
private void MyEventHandler() {
// some code
}
}
该类创建一个名为TheObject的SomeObject类型的内部对象,并为该对象上的事件添加一个事件处理程序.
由于TheObject是一个公共属性,这意味着任何其他代码都可以维护指向该对象的指针;反过来,这将使MyClass类型的对象保持活动状态,因为TheObject具有以事件处理程序的形式指向MyClass的指针.
因此,我认为保持此代码不受此事件影响的唯一方法是向MyClass添加终结器:
public ~MyClass() {
TheObject?.MyEvent -= MyEventHandler;
}
这太糟糕了,因为终结器会将MyClass类型的对象提升到下一代GC生成,但我是否认为这是避免这种潜在内存泄漏的唯一方法?
解决方法:
您的解决方案实际上不会解决问题,因为在收集对象之前不会调用终结器本身,并且正确识别TheObject将通过事件处理程序使对象保持活动状态.
有两个可能的修复:
>使MyClass实现IDisposable并在Dispose方法中取消注册事件处理程序. C#具有使用语法来帮助使用该类
>对事件使用弱引用,而不依赖于默认事件语法.
一个简单的实现是:
public interface ISubscription
{
bool IsAlive { get; }
void Fire();
}
public class Subscrition<T> : ISubscription
where T: class
{
public Subscrition(T target, Action<T> fire)
{
this.Target = new WeakReference<T>(target);
this.FireHandler = fire;
}
public WeakReference<T> Target { get; }
public Action<T> FireHandler { get; }
public bool IsAlive => this.Target.TryGetTarget(out var t);
public void Fire()
{
if (this.Target.TryGetTarget(out var target))
{
this.FireHandler(target);
}
}
}
public class WeakEvent
{
List<ISubscription> weakHandlers = new List<ISubscription>();
public void Register<T>(T target, Action<T> fire)
where T:class
{
this.Prune();
this.weakHandlers.Add(new Subscrition<T>(target, fire));
}
public void Unregister(ISubscription subscription)
{
this.Prune();
this.weakHandlers.Remove(subscription);
}
// Prune any dead handlers.
public void Prune()
{
this.weakHandlers.RemoveAll(_ => !_.IsAlive);
}
public void Fire()
{
this.Prune();
this.weakHandlers.ForEach(_ => _.Fire());
}
}
用法:
public class SomeObject
{
public WeakEvent WeakMyEvent = new WeakEvent();
}
public class MyClass
{
public SomeObject TheObject { get; }
public MyClass()
{
TheObject = new SomeObject();
TheObject.WeakMyEvent.Register(this, t => t.MyEventHandler());
}
private void MyEventHandler()
{
// some code
}
}
您还可以签出this article以获得更复杂的实施
内容总结
以上是互联网集市为您收集整理的C#:如何避免这种潜在的内存泄漏全部内容,希望文章能够帮你解决C#:如何避免这种潜在的内存泄漏所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。