如何重构嵌套使用中涉及的代码
本文关键字:代码 嵌套 何重构 重构 | 更新日期: 2023-09-27 18:32:04
我有一些代码有很多重复。 问题来自我正在处理嵌套IDisposable
类型的事实。 今天我有一些看起来像:
public void UpdateFromXml(Guid innerId, XDocument someXml)
{
using (var a = SomeFactory.GetA(_uri))
using (var b = a.GetB(_id))
using (var c = b.GetC(innerId))
{
var cWrapper = new SomeWrapper(c);
cWrapper.Update(someXml);
}
}
public bool GetSomeValueById(Guid innerId)
{
using (var a = SomeFactory.GetA(_uri))
using (var b = a.GetB(_id))
using (var c = b.GetC(innerId))
{
return c.GetSomeValue();
}
}
对于这些方法中的每一个,整个嵌套using
块都是相同的(显示了两个,但大约有十个)。 唯一不同的是,当您到达using
块的内部级别时会发生什么。
我想的一种方法是做一些类似的事情:
public void UpdateFromXml(Guid innerId, XDocument someXml)
{
ActOnC(innerId, c =>
{
var cWrapper = new SomeWrapper(c);
cWrapper.Update(someXml);
});
}
public bool GetSomeValueById(Guid innerId)
{
var result = null;
ActOnC(innerId, c => { result = c.GetSomeValue(); });
return result;
}
private void ActOnC(Guid innerId, Action<TheCType> action)
{
using (var a = SomeFactory.GetA(_uri))
using (var b = a.GetB(_id))
using (var c = b.GetC(innerId))
{
action(c);
}
}
这有效,只是解析起来有点笨拙(作为人类)。 有没有人对如何减少像这样的嵌套using
块周围的代码重复有任何其他建议? 如果它们不是IDisposable
那么人们可能只是创建一个方法来返回b.GetC(innerId)
的结果......但这里的情况并非如此。
我喜欢BFree提供的答案作为开始,但我会做一些修改。
//Give it a better name; this isn't designed to be a general purpose class
public class MyCompositeDisposable : IDisposable
{
public MyCompositeDisposable (string uri, int id, int innerid)
{
A = SomeFactory.GetA(uri);
B = A.GetB(id);
C = B.GetC(innerId);
}
//You can make A & B private if appropriate;
//not sure if all three or just C should be exposed publicly.
//Class names are made up; you'll need to fix.
//They should also probably be given more meaningful names.
public ClassA A{get;private set;}
public ClassB B{get;private set;}
public ClassC C{get;private set;}
public void Dispose()
{
A.Dispose();
B.Dispose();
C.Dispose();
}
}
完成此操作后,您可以执行以下操作:
public bool GetSomeValueById(Guid innerId)
{
using(MyCompositeDisposable d = new MyCompositeDisposable(_uri, _id, innerId))
{
return d.C.GetSomeValue();
}
}
请注意,MyCompositeDisposable 可能需要在构造函数和 Dispose 方法中包含 try/finally 块,以便正确创建/销毁中的错误确保最终没有未释放的内容。
在 Rx 框架中,有一个名为 CompositeDisposable
http://msdn.microsoft.com/en-us/library/system.reactive.disposables.compositedisposable%28v=vs.103%29.aspx 的类
滚动应该不会太难(尽管非常精简的版本):
public class CompositeDisposable : IDisposable
{
private IDisposable[] _disposables;
public CompositeDisposable(params IDisposable[] disposables)
{
_disposables = disposables;
}
public void Dispose()
{
if(_disposables == null)
{
return;
}
foreach(var disposable in _disposables)
{
disposable.Dispose();
}
}
}
然后这看起来更干净一点:
public void UpdateFromXml(Guid innerId, XDocument someXml)
{
var a = SomeFactory.GetA(_uri);
var b = a.GetB(_id);
var c = b.GetC(innerId);
using(new CompositeDisposable(a,b,c))
{
var cWrapper = new SomeWrapper(c);
cWrapper.Update(someXml);
}
}
您始终可以创建一个更大的上下文来管理应创建/释放哪些对象。 然后编写一个方法来创建更大的上下文...
public class DisposeChain<T> : IDisposable where T : IDisposable
{
public T Item { get; private set; }
private IDisposable _innerChain;
public DisposeChain(T theItem)
{
this.Item = theItem;
_innerChain = null;
}
public DisposeChain(T theItem, IDisposable inner)
{
this.Item = theItem;
_innerChain = inner;
}
public DisposeChain<U> Next<U>(Func<T, U> getNext) where U : IDisposable
{
try
{
U nextItem = getNext(this.Item);
DisposeChain<U> result = new DisposeChain<U>(nextItem, this);
return result;
}
catch //an exception occurred - abort construction and dispose everything!
{
this.Dispose()
throw;
}
}
public void Dispose()
{
Item.Dispose();
if (_innerChain != null)
{
_innerChain.Dispose();
}
}
}
然后使用它:
public DisposeChain<DataContext> GetCDisposeChain()
{
var a = new DisposeChain<XmlWriter>(XmlWriter.Create((Stream)null));
var b = a.Next(aItem => new SqlConnection());
var c = b.Next(bItem => new DataContext(""));
return c;
}
public void Test()
{
using (var cDisposer = GetCDisposeChain())
{
var c = cDisposer.Item;
//do stuff with c;
}
}
如果您的Dispoable
类型正确处理了所有一次性成员,则只需要一个 using 语句。
例如,这个:
public bool GetSomeValueById(Guid innerId)
{
using (var a = SomeFactory.GetA(_uri))
using (var b = a.GetB(_id))
using (var c = b.GetC(innerId))
{
return c.GetSomeValue();
}
}
如果 A 具有类型 B 和 C 的成员,并且 A 在其处置方法中处理了 B 和 C,则可以变成这样:
public bool GetSomeValueById(Guid innerId)
{
using (var a = SomeFactory.GetA(_uri))
{
return a.GetSomeValue();
}
}
class A : IDisposable
{
private a;
private b;
public A (B b, C c)
{
this.b = b; this.c = c;
}
public void Dispose()
{
Dispose(true);
}
protected void Dispose(bool disposing)
{
if (disposing)
{
b.Dispose();
c.Dispose();
}
}
}
但是,您必须修改工厂以将 b 和 c 注入 a。