В настоящее время я изучаю, как использовать Autofac, и я застрял с удалением IDisposable
объектов детерминистически. Позвольте мне сначала представить ситуацию, прежде чем я изложу свою проблему.
Исходная позиция:
Допустим, моя объектная модель определяется через следующие интерфейсы:
interface IApple : IDisposable
{
void Consume();
}
interface IHorse
{
void Eat(IApple apple); // is supposed to call apple.Consume()
}
interface IHorseKeeper
{
void FeedHorse(); // is supposed to call horse.Eat(apple)
// where 'horse' is injected into IHorseKeeper
// and 'apple' is generated by IHorseKeeper on-the-fly
}
Далее я определяю делегата, который будет использоваться как IApple
фабрика:
delegate IApple AppleFactory;
Конфигурация Autofac:
Теперь я бы зарегистрировал перечисленные выше типы следующим образом - обратите внимание, что я опускаю код обоих классов Apple
и Horse
, так как они тривиальны для реализации:
var builder = new Autofac.ContainerBuilder();
builder.RegisterType<Apple>().As<IApple>();
builder.RegisterType<Horse>().As<IHorse>();
builder.RegisterType<HorseKeeper>().As<IHorseKeeper>();
builder.RegisterGeneratedFactory<AppleFactory>();
Моя проблема:
Я не совсем знаю, как реализовать метод IHorseKeeper.Feed
. Вот что у меня сейчас есть:
class HorseKeeper : IHorseKeeper
{
private readonly IHorse horse;
private readonly AppleFactory appleFactory;
public HorseKeeper(IHorse horse, AppleFactory appleFactory)
// ^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^^^^^^^^
// constructor injection
{
this.horse = horse;
this.appleFactory = appleFactory;
}
public void FeedHorse()
{
using (var apple = appleFactory())
{
horse.Eat(apple);
} // <- Dispose() apple now (ASAP), as it's no longer needed!
}
}
Это тот код, который я хотел бы получить, так как он полностью независим от автофака. Он также может работать с другим контейнером IoC, если AppleFactory
работает как положено.
Однако, поскольку Autofac обрабатывает AppleFactory
для меня, он будет отслеживать все IApple
объектов, которые он производит для меня, и поэтому захочет Dispose
их самостоятельно в конце срока службы контейнера. Т.е. произведенный apple
будет утилизирован дважды.
Полагаю, регистрация IApple
в качестве .ExternallyOwned()
не является жизнеспособным решением, так как могут быть случаи, когда Autofac легче обрабатывать время жизни IApple
s.
Детерминированное удаление с помощью Autofac требует создания вложенного контейнера с использованием container.BeginLifetimeScope()
, однако я не хочу использовать это внутри HorseKeeper.FeedHorse
, потому что тогда HorseKeeper
становится зависимым от Autofac, и я хотел бы сохранить мой код IoC-агностиком.
Вопрос:
Как мне реализовать HorseKeeper.FeedHorse
в IoC (Autofac) -агностическом способе, гарантируя, что сгенерированные на лету объекты располагаются должным образом?