Является ли хорошей практикой передавать ядро Ninject?
Я пишу небольшой фреймворк, который выполнил пару задач. Некоторым из этих задач нужны особые свойства, которые вводятся через Ninject.
Допустим, у меня есть следующий конструктор в моем базовом классе, который представляет одну задачу:
protected DDPSchedulerTask(ILogger logger, List<string> platforms, IBackOfficeDataStore backOfficeDataStore, ICommonDataStore commonDataStore)
{
_logger = logger;
_platforms = platforms;
_backOfficeDataStore = backOfficeDataStore;
_commonDataStore = commonDataStore;
}
Эти свойства необходимы для всех задач, поэтому я внедряю их, используя Ninject со следующим модулем Ninject.
public class DDPDependencyInjectionBindingConfiguration : NinjectModule
{
#region NinjectModule Members
/// <summary>
/// Loads the module into the kernel.
/// </summary>
public override void Load()
{
Bind<Scheduler>().ToSelf(); // Make sure that the 'Scheduler' is resolved to itself.
Bind<ILogger>().ToMethod(context => LogFactory.Create()); // Make sure that an instance of an ILogger is created through the LogFactory.
// Straightforward binding.
Bind<ICommonDataStore>().To<Common>();
Bind<IBackOfficeDataStore>().To<BtDbInteract>();
Bind<IDirectoryResolver>().To<Demo>();
}
#endregion
}
Мой объект планировщика сам по себе, если первая запись в цепочке, которая должна быть разрешена Ninject, поэтому я решаю это вручную через Ninject.
var schedulerInstance = kernel.Get<Scheduler>();
Теперь у моего планировщика есть метод, который добавляет задачи в список, а не с помощью Ninject:
var tasksList = new List<DDPSchedulerTask>
{
new AWNFileGeneratorTask(_logger, availablePlatforms, _backOfficeDataStore, _commonDataStore)
};
Затем все эти задачи выполняются. Теперь некоторые из этих задач требуют дополнительных зависимостей, которые я хотел бы решить с помощью Ninject, но как мне это сделать?
Внутри задачи я создал свойство с Inject
атрибут, но объект остается нулевым.
[Inject]
private IDirectoryResolver directoryResolver { get; set; }
У кого-нибудь есть идеи о том, как это можно решить?
Я могу передать ядро различным задачам, но что-то подсказывает мне, что это неправильный подход.
С уважением
2 ответа
В таких случаях я обычно использую шаблон Factory. В планировщике вы можете использовать фабрику задач в качестве зависимости. Эта фабрика также может иметь несколько методов для создания различных типов задач.
class DDPSchedulerTaskFactory : IDDPSchedulerTaskFactory
{
DDPSchedulerTaskFactory(ILogger logger, List<string> platforms, IBackOfficeDataStore backOfficeDataStore, ICommonDataStore commonDataStore)
{
_logger = logger;
_platforms = platforms;
_backOfficeDataStore = backOfficeDataStore;
_commonDataStore = commonDataStore;
}
public IDDPSchedulerTask CreateNormal(){
return new DDPSchedulerTask(
_logger,
_platforms,
_backOfficeDataStore,
_commonDataStore);
}
public IDDPSchedulerTask CreateSpecial(someAdditionalParameter){
return new AnotherDDPSchedulerTask(
_logger,
_platforms,
_backOfficeDataStore,
_commonDataStore,
someAdditionalParameter);
}
public IDDPSchedulerTask CreateTaskWithDifferentDependenties(yetAnotherParameter){
return new AnotherDDPSchedulerTask(
_logger,
_platforms,
yetAnotherParameter);
}
}
Чем в вашем планировщике вы можете добавить такие задачи:
class Scheduler{
IDDPSchedulerTaskFactory _taskFactory;
public Scheduler(IDDPSchedulerTaskFactory taskFactory){
_taskFactory = taskFactory; // factory gets all the needed dependencies for all tasks from DI
}
...
public void ConfigureTasks(){
_tasks.Add(_taskFactory.CreateNormal());
_tasks.Add(_taskFactory.CreateSpecial("Some important message"));
_tasks.Add(_taskFactory.CreateTaskWithDifferentDependenties(123));
}
}
Вы должны воспользоваться https://github.com/ninject/Ninject.Extensions.Factory.
Просто создайте интерфейс, который представляет ваши фабрики задач. Затем передайте информацию Ninject, которая является фабрикой, и он создаст для вас полные реализации этого интерфейса.
using System;
using System.Collections.Generic;
using Ninject;
using Ninject.Extensions.Factory;
internal class Program
{
public static void Main(string[] args)
{
IKernel ninjectKernel = new StandardKernel();
ninjectKernel.Bind<DDPSchedulerTask>().ToSelf();
ninjectKernel.Bind<AWNFileGeneratorTask>().ToSelf();
ninjectKernel.Bind<IDirectoryResolver>().To<DirectoryResolver>();
ninjectKernel.Bind<ITaskFactory>().ToFactory();
var mainTask = ninjectKernel.Get<DDPSchedulerTask>();
mainTask.CreateDbSchedulerTask();
mainTask.CreateAwnFileTask();
Console.ReadLine();
}
}
public interface ITaskFactory
{
TTask CreateTask<TTask>() where TTask : DDPSchedulerTask;
}
public class DDPSchedulerTask
{
private readonly ITaskFactory _tasksFactory;
private readonly List<DDPSchedulerTask> _tasksList;
public DDPSchedulerTask(ITaskFactory tasksFactory)
{
_tasksFactory = tasksFactory;
_tasksList = new List<DDPSchedulerTask>();
}
public void CreateAwnFileTask()
{
var task = _tasksFactory.CreateTask<AWNFileGeneratorTask>();
_tasksList.Add(task);
}
public void CreateDbSchedulerTask()
{
var task = _tasksFactory.CreateTask<DDPSchedulerTask>();
_tasksList.Add(task);
}
}
public class AWNFileGeneratorTask : DDPSchedulerTask
{
[Inject]
public IDirectoryResolver DirectoryResolver { get; set; }
public AWNFileGeneratorTask(ITaskFactory tasksFactory)
: base(tasksFactory)
{
}
}
public interface IDirectoryResolver
{
}
public class DirectoryResolver : IDirectoryResolver
{
}
@gisek Как уже отмечалось, внедрение зависимости Гисеке через свойство - не лучшее решение. В этом примере вы также можете использовать инъекцию конструктора.
public class AWNFileGeneratorTask : DDPSchedulerTask
{
private IDirectoryResolver _directoryResolver;
public AWNFileGeneratorTask(ITaskFactory tasksFactory, IDirectoryResolver directoryResolver)
: base(tasksFactory)
{
_directoryResolver = directoryResolver;
}
}
Дополнительные параметры ввода:
public interface ITaskFactory
{
DDPSchedulerTask CreateDDPSchedulerTask();
AWNFileGeneratorTask CreateAWNFileGeneratorTask(string extraParam);
}
public class AWNFileGeneratorTask : DDPSchedulerTask
{
private IDirectoryResolver _directoryResolver;
private string _extraParam;
public AWNFileGeneratorTask(ITaskFactory tasksFactory, IDirectoryResolver directoryResolver,
string extraParam)
: base(tasksFactory)
{
_extraParam = extraParam;
_directoryResolver = directoryResolver;
}
}
public void CreateAwnFileTask()
{
var task = _tasksFactory.CreateAWNFileGeneratorTask("extra");
_tasksList.Add(task);
}