ข้อจำกัดความรับผิดชอบ:ณ ต้นปี 2558 มีการเปรียบเทียบคุณสมบัติ IoC Container จากJimmy Bogard ที่ยอดเยี่ยมนี่เป็นบทสรุป:
เปรียบเทียบภาชนะบรรจุ:
- Autofac
- Ninject
- หัวฉีดง่าย
- StructureMap
- เอกภาพ
- วินด์เซอร์
สถานการณ์คือ: ฉันมีส่วนต่อประสาน, IMediator, ซึ่งฉันสามารถส่งคำขอ / ตอบกลับเดียวหรือการแจ้งเตือนไปยังผู้รับหลายคน:
public interface IMediator
{
TResponse Send<TResponse>(IRequest<TResponse> request);
Task<TResponse> SendAsync<TResponse>(IAsyncRequest<TResponse> request);
void Publish<TNotification>(TNotification notification)
where TNotification : INotification;
Task PublishAsync<TNotification>(TNotification notification)
where TNotification : IAsyncNotification;
}
ฉันสร้างชุดคำขอ / ตอบกลับ / การแจ้งเตือนพื้นฐาน:
public class Ping : IRequest<Pong>
{
public string Message { get; set; }
}
public class Pong
{
public string Message { get; set; }
}
public class PingAsync : IAsyncRequest<Pong>
{
public string Message { get; set; }
}
public class Pinged : INotification { }
public class PingedAsync : IAsyncNotification { }
ฉันสนใจที่จะดูบางสิ่งเกี่ยวกับการสนับสนุนคอนเทนเนอร์สำหรับข้อมูลทั่วไป:
- ตั้งค่าสำหรับ generics แบบเปิด (การลงทะเบียน IRequestHandler <,> ได้อย่างง่ายดาย)
- การตั้งค่าสำหรับการลงทะเบียนหลายครั้งของ generics แบบเปิด (INotificationHandlers สองคนขึ้นไป)
การตั้งค่าสำหรับความแปรปรวนทั่วไป (การลงทะเบียนตัวจัดการสำหรับการอ้างอิงพื้นฐาน / การสร้างท่อส่งคำขอ) ตัวจัดการของฉันค่อนข้างตรงไปตรงมาพวกเขาเพียงแค่ส่งออกไปยังคอนโซล:
public class PingHandler : IRequestHandler<Ping, Pong> { /* Impl */ }
public class PingAsyncHandler : IAsyncRequestHandler<PingAsync, Pong> { /* Impl */ }
public class PingedHandler : INotificationHandler<Pinged> { /* Impl */ }
public class PingedAlsoHandler : INotificationHandler<Pinged> { /* Impl */ }
public class GenericHandler : INotificationHandler<INotification> { /* Impl */ }
public class PingedAsyncHandler : IAsyncNotificationHandler<PingedAsync> { /* Impl */ }
public class PingedAlsoAsyncHandler : IAsyncNotificationHandler<PingedAsync> { /* Impl */ }
Autofac
var builder = new ContainerBuilder();
builder.RegisterSource(new ContravariantRegistrationSource());
builder.RegisterAssemblyTypes(typeof (IMediator).Assembly).AsImplementedInterfaces();
builder.RegisterAssemblyTypes(typeof (Ping).Assembly).AsImplementedInterfaces();
- ยาชื่อสามัญแบบเปิด: ใช่โดยปริยาย
- มี generics แบบเปิดหลายตัว: ใช่โดยปริยาย
- ความแตกต่างทั่วไป: ใช่อย่างชัดเจน
Ninject
var kernel = new StandardKernel();
kernel.Components.Add<IBindingResolver, ContravariantBindingResolver>();
kernel.Bind(scan => scan.FromAssemblyContaining<IMediator>()
.SelectAllClasses()
.BindDefaultInterface());
kernel.Bind(scan => scan.FromAssemblyContaining<Ping>()
.SelectAllClasses()
.BindAllInterfaces());
kernel.Bind<TextWriter>().ToConstant(Console.Out);
- ยาชื่อสามัญแบบเปิด: ใช่โดยปริยาย
- มี generics แบบเปิดหลายตัว: ใช่โดยปริยาย
- ความแตกต่างทั่วไป: ใช่ด้วยส่วนขยายที่ผู้ใช้สร้างขึ้น
หัวฉีดง่าย
var container = new Container();
var assemblies = GetAssemblies().ToArray();
container.Register<IMediator, Mediator>();
container.Register(typeof(IRequestHandler<,>), assemblies);
container.Register(typeof(IAsyncRequestHandler<,>), assemblies);
container.RegisterCollection(typeof(INotificationHandler<>), assemblies);
container.RegisterCollection(typeof(IAsyncNotificationHandler<>), assemblies);
- ข้อมูลทั่วไปแบบเปิด: ใช่อย่างชัดเจน
- หลาย generics เปิด: ใช่อย่างชัดเจน
- ความแตกต่างทั่วไป: ใช่โดยนัย (พร้อมอัปเดต 3.0)
StructureMap
var container = new Container(cfg =>
{
cfg.Scan(scanner =>
{
scanner.AssemblyContainingType<Ping>();
scanner.AssemblyContainingType<IMediator>();
scanner.WithDefaultConventions();
scanner.AddAllTypesOf(typeof(IRequestHandler<,>));
scanner.AddAllTypesOf(typeof(IAsyncRequestHandler<,>));
scanner.AddAllTypesOf(typeof(INotificationHandler<>));
scanner.AddAllTypesOf(typeof(IAsyncNotificationHandler<>));
});
});
- ข้อมูลทั่วไปแบบเปิด: ใช่อย่างชัดเจน
- หลาย generics เปิด: ใช่อย่างชัดเจน
- ความแตกต่างทั่วไป: ใช่โดยปริยาย
เอกภาพ
container.RegisterTypes(AllClasses.FromAssemblies(typeof(Ping).Assembly),
WithMappings.FromAllInterfaces,
GetName,
GetLifetimeManager);
/* later down */
static bool IsNotificationHandler(Type type)
{
return type.GetInterfaces().Any(x => x.IsGenericType && (x.GetGenericTypeDefinition() == typeof(INotificationHandler<>) || x.GetGenericTypeDefinition() == typeof(IAsyncNotificationHandler<>)));
}
static LifetimeManager GetLifetimeManager(Type type)
{
return IsNotificationHandler(type) ? new ContainerControlledLifetimeManager() : null;
}
static string GetName(Type type)
{
return IsNotificationHandler(type) ? string.Format("HandlerFor" + type.Name) : string.Empty;
}
- ยาชื่อสามัญแบบเปิด: ใช่โดยปริยาย
- Generics แบบเปิดหลายตัว: ใช่พร้อมส่วนขยายที่ผู้ใช้สร้างขึ้น
- ความแตกต่างทั่วไป: derp
วินด์เซอร์
var container = new WindsorContainer();
container.Register(Classes.FromAssemblyContaining<IMediator>().Pick().WithServiceAllInterfaces());
container.Register(Classes.FromAssemblyContaining<Ping>().Pick().WithServiceAllInterfaces());
container.Kernel.AddHandlersFilter(new ContravariantFilter());
- ยาชื่อสามัญแบบเปิด: ใช่โดยปริยาย
- มี generics แบบเปิดหลายตัว: ใช่โดยปริยาย
- ความแตกต่างทั่วไป: ใช่ด้วยส่วนขยายที่ผู้ใช้สร้างขึ้น