【问题标题】:Is there a way to have a using statement with dynamically-created targets?有没有办法使用动态创建目标的 using 语句?
【发布时间】:2023-04-08 18:48:02
【问题描述】:

假设我定义了一个类MyDisposable : IDisposable。我知道我可以为using 语句提供一个硬编码的IDisposable 对象列表:

using (MyDisposable firstDisposable = new MyDisposable(),
    secondDisposable = new MyDisposable())
{
    // do something
}

现在假设我有一些方法可以对一次性对象的集合执行操作,我想在 using 语句中执行此操作。它可能看起来像这样(但这当然不起作用,因为 using 块需要一个或多个 IDisposable 对象,而我正在传递一个集合对象):

using (var myDisposables = GetMyDisposables())
{
    foreach (var myDisposable in myDisposables)
    {
        DoSomething(myDisposable);
        DoSomethingElse(myDisposable);
    }
}

为了清楚起见,以下是其他方法:

static List<MyDisposable> GetMyDisposables()
{
    throw new NotImplementedException(); // return a list of MyDisposable objects
}

static void DoSomething(MyDisposable withMyDisposable)
{
    // something
}

static void DoSomethingElse(MyDisposable withMyDisposable)
{
    // something else
}

有没有什么方法可以用using 语句完成这个任务?还是我只需要丢弃该语句并手动处理?

【问题讨论】:

  • 你可以创建一个DisposableCollection&lt;T&gt; : IDisposable, IEnumerable&lt;T&gt; where T : IDisposable
  • 您可以编写IDisposable,它封装了您的MyDisposable 对象的集合,并在您处置它时处置其所有元素。
  • 为什么不在 foreach 块中添加 using 语句?
  • @peinearydevelopment -- 因为那样我就不能使用 next foreach 中的对象...它们已经在第一个中被处理了。
  • @peinearydevelopment 因为这样处理项目的错误会泄漏集合中所有尚未处理的项目。

标签: c# .net idisposable using-statement


【解决方案1】:

您可以采取的一种方法是创建IDisposable 对象的集合,这也是IDisposable

class CollectionOfDisposable<T> : IDisposable where T : IDisposable  {
    public IList<T> Members {get; private set;}
    public CollectionOfDisposable(IEnumerable<T> members) {
        Members = members.ToList();
    }
    public void Dispose() {
        var exceptions = new List<Exception>();
        foreach (var item in Members) {
            try {
                item.Dispose();
            } catch (Exception e) {
                exceptions.Add(e);
            }
        }
        if (exceptions.Count != 0) {
            throw new AggregateException(exceptions);
        }
    }
}

现在你可以这样写了:

using (var myDisposables = GetMyDisposables()) {
    foreach (var myDisposable in myDisposables.Members) {
        DoSomething(myDisposable);
        DoSomethingElse(myDisposable);
    }
}

static CollectionOfDisposable<MyDisposable> GetMyDisposables() {
    throw new NotImplementedException(); // return a list of MyDisposable objects
}

【讨论】:

【解决方案2】:

您必须创建自己的类型来实现 IDisposable 并在其构造函数中接受一次性对象的集合,保留它们并在处置时将它们全部处置。

【讨论】:

    猜你喜欢
    • 2020-09-22
    • 2014-06-11
    • 1970-01-01
    • 2011-04-30
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2022-10-12
    • 1970-01-01
    相关资源
    最近更新 更多