Хабр Курсы для всех
РЕКЛАМА
Практикум, Хекслет, SkyPro, авторские курсы — собрали всех и попросили скидки. Осталось выбрать!
Уже на этом этапе в голове начинает звенеть тревожный звоночек: а не слишком ли много зависимостей у нас внедряется в класс Worker?
Не будет правильнее в этом случае воспользоваться фасадом и предоставить клиенту ограниченный набор методов а не сами объекты репозиториев?
Знаю что можно определить свой атрибут который будет работать подобно [Inject], есть ли что то подобное для [Intercept]
public class CustomPlanningStrategy<TAttribute, TInterceptor> :
NinjectComponent, IPlanningStrategy
where TAttribute : Attribute
where TInterceptor : IInterceptor
{
private readonly IAdviceFactory _adviceFactory;
private readonly IAdviceRegistry _adviceRegistry;
public CustomPlanningStrategy(
IAdviceFactory adviceFactory, IAdviceRegistry adviceRegistry)
{
_adviceFactory = adviceFactory;
_adviceRegistry = adviceRegistry;
}
public void Execute(IPlan plan)
{
var methods = GetCandidateMethods(plan.Type);
foreach (var method in methods)
{
var attributes = method.GetCustomAttributes(typeof (TAttribute), true) as TAttribute[];
if (attributes != null && attributes.Length == 0)
continue;
var advice = _adviceFactory.Create(method);
advice.Callback = request => request.Kernel.Get<TInterceptor>();
_adviceRegistry.Register(advice);
if (!plan.Has<ProxyDirective>())
plan.Add(new ProxyDirective());
}
}
private static IEnumerable<MethodInfo> GetCandidateMethods(Type type)
{
var methods = type.GetMethods(
BindingFlags.Public |
BindingFlags.NonPublic |
BindingFlags.Instance
);
return methods.Where(ShouldIntercept);
}
private static bool ShouldIntercept(MethodInfo methodInfo)
{
return methodInfo.DeclaringType != typeof (object) &&
!methodInfo.IsPrivate &&
!methodInfo.IsFinal;
}
}
IKernel kernel = new StandardKernel(new CommonModule());
kernel.Components.Add<IPlanningStrategy,
CustomPlanningStrategy<LogExceptionAttribute, ExceptionInterceptor>>();
Worker worker = kernel.Get<Worker>();
public class LogExceptionAttribute : Attribute
{
}
если в приложении 200 классов используют DI, то при попытке получения экземпляра класса, который находится на вершине дерева зависимостей, будет создано 200 экземпляров остальных классов, даже если в текущем сценарии будет использовано 10а если прописать время жизни объекта то будет всё в порядке. Singleton, PerThread, PerRequest, PerScope обычно DI фреймворки предоставляют стандартный набор для указания времени жизни. По моему это один из смыслов использовать DI. К тому же как вы работаете с EF? создавали контекст для каждого репозитория свой?
К тому же как вы работаете с EF? создавали контекст для каждого репозитория свой?
Property Injection не пробовали? Конструктор не такой страшный и не знаю как в Ninject, но в AutoFac и Unity циклические зависимости тоже отлавливаются. Но мне больше нравиться когда все зависимости прописаны в конструкторе, напрягает только если есть наследование.
Advanced Dependency Injection на примере Ninject