- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
好的,让我们更笼统地重新表述这个问题。
有没有一种方法可以使用反射在运行时动态调用您可能正在覆盖的基类方法。您不能在编译时使用 'base' 关键字,因为您不能确定它是否存在。在运行时,我想列出我的祖先方法并调用祖先方法。
我尝试使用 GetMethods() 等,但它们返回的只是指向该方法的最派生实现的“指针”。不是基类的实现。
我们正在用 C# 3.0 开发一个具有相对较大类层次结构的系统。其中一些类,在层次结构中的任何位置,都有需要的资源处置,那些实现了IDisposable接口(interface)。
现在,为了方便代码的维护和重构,我想找到一种方法,用于实现 IDisposable 的类,如果任何祖先也实现了 IDisposable,则“自动”调用 base.Dispose(bDisposing)。这样,如果层次结构中较高的某个类开始实现或停止实现将自动处理的 IDisposable。
问题有两个方面。
第一部分,找到实现 IDisposable 的祖先,我已经能够处理了。
第二部分是棘手的部分。尽管我所有的努力,我无法从派生类调用 base.Dispose(bDisposing)。我所有的尝试都失败了。他们要么造成编译错误或调用了错误的 Dispose() 方法,即最派生的方法,从而永远循环。
主要问题是,如果没有诸如实现它的祖先(请注意,可能还没有实现 IDisposable 的祖先,但我希望派生代码在这种情况下准备就绪将来会发生一件事)。这给我们留下了 Reflection 机制,但我没有找到合适的方法。我们的代码充满了先进的反射技术,我想我没有遗漏任何明显的东西。
我最好的办法是在注释代码中使用一些条件代码。更改 IDisposable 层次结构会破坏构建(如果不存在 IDisposable 祖先)或抛出异常(如果存在 IDisposable 祖先但未调用 base.Dispose)。
这是我发布的一些代码,向您展示我的 Dispose(bDisposing) 方法是什么样的。我将这段代码放在所有 Dispose() 的末尾整个层次结构中的方法。任何新类都是从也包含此代码的模板创建的。
public class MyOtherClassBase
{
// ...
}
public class MyDerivedClass : MyOtherClassBase, ICalibrable
{
private bool m_bDisposed = false;
~MyDerivedClass()
{
Dispose(false);
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
protected virtual void Dispose(bool bDisposing)
{
if (!m_bDisposed) {
if (bDisposing) {
// Dispose managed resources
}
// Dispose unmanaged resources
}
m_bDisposed = true;
Type baseType = typeof(MyDerivedClass).BaseType;
if (baseType != null) {
if (baseType.GetInterface("IDisposable") != null) {
// If you have no ancestors implementing base.Dispose(...), comment
// the following line AND uncomment the throw.
//
// This way, if any of your ancestors decide one day to implement
// IDisposable you will know about it right away and proceed to
// uncomment the base.Dispose(...) in addition to commenting the throw.
//base.Dispose(bDisposing);
throw new ApplicationException("Ancestor base.Dispose(...) not called - "
+ baseType.ToString());
}
}
}
}
那么,我想问的是有没有一种方法可以改为自动/有条件地调用 base.Dispose()?
应用程序中还有另一种机制,其中所有对象都注册到一个主类。该类检查它们是否实现了 IDisposable。如果是这样,它们将由应用程序正确处理。这避免了让代码使用类来处理自己到处调用 Dispose() 。因此,将 IDisposable 添加到没有 IDisposable 祖先历史的类中仍然可以完美地工作。
最佳答案
标准模式是让您的基类实现 IDisposable 和非虚拟 Dispose() 方法,并实现虚拟 Dispose(bool) 方法,那些持有一次性资源的类必须重写该方法。他们应该总是调用他们的基础 Dispose(bool) 方法,该方法最终将链接到层次结构中的顶级类。只有那些覆盖它的类才会被调用,所以链通常很短。
终结器,在 C# 中拼写为 ~Class:不要。很少有类需要一个,并且很容易意外地保留大对象图,因为终结器在释放内存之前至少需要两次集合。在对象不再被引用后的第一个集合中,它被放入要运行的终结器队列中。它们在单独的专用线程上运行,该线程仅运行终结器(如果它被阻塞,则不再运行终结器并且您的内存使用量激增)。一旦终结器运行,收集适当世代的下一个集合将释放该对象以及它所引用但未被引用的任何其他对象。不幸的是,因为它在第一次收集中幸存下来,所以它将被放入收集频率较低的老年代。因此,您应该尽早并经常处置。
通常,您应该实现一个小型资源包装器类,仅 管理资源生命周期并在该类上实现终结器,外加 IDisposable。然后,该类的用户应该在处理它时调用 Dispose。不应有指向用户的反向链接。这样,只有真正需要终结的东西才会出现在终结队列中。
如果您将在层次结构中的任何地方需要它们,实现 IDisposable 的基类应该实现终结器并调用 Dispose(bool),将 false 作为参数传递。
针对 Windows Mobile 开发人员(VS2005 和 2008、.NET Compact Framework 2.0 和 3.5)的警告:您放置在设计器界面上的许多非控件,例如菜单栏、计时器、HardwareButtons 派生自 System.ComponentModel.Component,它实现了终结器。对于桌面项目,Visual Studio 将组件添加到名为 components
的 System.ComponentModel.Container,它会在处理表单时生成要处理的代码 - 它会依次处理所有已添加的组件。对于移动项目,会生成 Dispose components
的代码,但是将组件放到表面上不会生成将其添加到 components
的代码>。调用 InitializeComponent 后,您必须在构造函数中自己执行此操作。
关于c# - 从派生类自动调用 base.Dispose(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/78141/
我在 dispose 和 finalize 之间犹豫不决。这是我的示例代码: public class Car:IDisposable { public string name
我有一个 Activity,我在其中创建并订阅 Single 类的多个实例(每个实例都在单独的后台线程中执行一些工作)。对于每个订阅,我将创建的 Disposable 实例添加到 CompositeD
我有一个服务、一个存储库和一个 DbContext。存储库拥有 DbContext,服务拥有存储库。 我的存储库应该实现 IDisposable 吗?如果是这样,我的服务是否也应该实现 IDispos
我有一个基于字符串列表处理文件流的应用程序,字符串可以是磁盘上的文件,也可以是 Zip 文件中的文件。为了清理代码,我想重构打开文件的过程。 我创建了一个返回文件内容 Stream 的方法,但是因为流
有没有一种方法可以从 Dispose 方法中取消处置,换句话说,在调用 A.Dispose() 时不处置 A。 更新我没有很好地解释,因为我想不出一个很好的方式来表达它。我拥有的是一个在单独线程中运行
如以下示例实现即 https://learn.microsoft.com/en-us/dotnet/standard/garbage-collection/implementing-dispose有一
我开始用 FXCop 来处理我的一些程序集,并注意到一些有趣的事情,我正试图全神贯注。 "Types that declare disposable members should also imple
当我创建表单时,自动生成的代码不包含重写的 Dispose 方法。这是否意味着没有为表单中的所有控件调用 Dispose? 最佳答案 当您在表单上调用 Dispose 时,它会为其 Controls
在我的 WPF 应用程序代码中,我收到以下警告: CA1001 Types that own disposable fields should be disposable Implement IDis
每当我读到 Close() 和 Dispose() 时,我都会看到很多只使用 Using Block 的引用资料,但我还没有找到如何在 WebMatrix C# Razor 语法中使用 Using B
我是 c# 和 VSTO 的新手。 我有一个 Excel 插件,我在其中创建了一个功能区和一些可以执行各种操作的新按钮。 我在 base.Dispose 中收到 NullReferenceExcept
我在 FCL 代码中发现了一个不寻常的样本。 这是 System.IO.BinaryReader 中的方法: protected virtual void Dispose(bool dispo
我正在尝试遵循以下教程并将其从 C# 转换为 Vb.net http://www.asp.net/mvc/overview/older-versions/getting-started-with-ef
我有一个 LINQ 对象,其中添加了一个附加方法。该类没有一次性属性或方法,但 FxCop 引发错误“拥有一次性字段的类型应该是一次性的”并引用该类。 到目前为止我已经减少了代码,但仍然收到错误: p
因此,默认的处理模式实现如下所示: class SomeClass : IDisposable { // Flag: Has Dispose already been called? bo
我在 WinForms 应用程序中使用 RxFramework。我正在尝试运行 Observable 异步并使用 CancellationDisposable 在用户单击按钮时取消操作。但它不工作!
有很多从 .NET 中的函数返回 IDisposable 对象数组或列表的示例。例如,Process.GetProcesses()。 如果我调用该方法,我是否有责任在遍历数组的所有成员时对它们进行 D
如果我用 C# 编写一个实现 IDisposable 的类,为什么我不能简单地实现 public void Dispose(){ ... } 处理释放任何非托管资源? 是 protected virt
LibGDX - 是否应该先执行 texture.dispose() 然后再执行 batch.dispose()?还是没有什么不同? 我有以下简单的代码。我的问题是 texture.dispose()
当应用程序收到来自服务的调用时,它会为每个调用打开一个表单。用户必须在每个窗口中进行输入并关闭它。为了让用户的工作顺利进行,我试图在显示下一个窗口时重新激活用户正在处理的窗口。 执行此操作的方法如下:
我是一名优秀的程序员,十分优秀!