如今在项目里大多都是直接使用微软的依赖注入框架,而微软的注入方式比较简单,不如 AutoFac 使用起来灵活,因而想给微软的依赖注入增长一些扩展,使得能够像AutoFac 同样比较灵活的注册服务html
将类型注册为其实现的接口,好比 pubic class UserService:IUserService,IUserRepository{}
,git
注册 UserService
为 IUserService
和 IUserRepository
github
等效于:框架
services.AddSingleton<IUserService, UserService>(); services.AddSingleton<IUserRepository, UserService>();
实现代码:this
/// <summary> /// RegisterTypeAsImplementedInterfaces /// </summary> /// <param name="services">services</param> /// <param name="type">type</param> /// <param name="serviceLifetime">service lifetime</param> /// <returns>services</returns> public static IServiceCollection RegisterTypeAsImplementedInterfaces(this IServiceCollection services, Type type, ServiceLifetime serviceLifetime = ServiceLifetime.Singleton) { if (type != null) { foreach (var interfaceType in type.GetImplementedInterfaces()) { services.Add(new ServiceDescriptor(interfaceType, type, serviceLifetime)); } } return services; }
在上面的基础上,加载指定程序集中的类型(不指定程序集则使用当前应用程序域中全部程序集),能够经过 typesFilter
来过滤要注册的类型插件
使用示例:3d
services.RegisterAssemblyTypesAsImplementedInterface(t => !t.IsAbstract && t.IsClass && t.IsAssignableTo<IEventHandler>(), typeof(NoticeViewEventHandler).Assembly);
实现源码:code
/// <summary> /// RegisterTypeAsImplementedInterfaces /// </summary> /// <param name="services">services</param> /// <param name="assemblies">assemblies</param> /// <returns>services</returns> public static IServiceCollection RegisterAssemblyTypesAsImplementedInterfaces(this IServiceCollection services, params Assembly[] assemblies) => RegisterAssemblyTypesAsImplementedInterfaces(services, typesFilter: null, ServiceLifetime.Singleton, assemblies); /// <summary> /// RegisterTypeAsImplementedInterfaces /// </summary> /// <param name="services">services</param> /// <param name="serviceLifetime">service lifetime</param> /// <param name="assemblies">assemblies</param> /// <returns>services</returns> public static IServiceCollection RegisterAssemblyTypesAsImplementedInterfaces(this IServiceCollection services, ServiceLifetime serviceLifetime, params Assembly[] assemblies) => RegisterAssemblyTypesAsImplementedInterfaces(services, typesFilter: null, serviceLifetime, assemblies); /// <summary> /// RegisterTypeAsImplementedInterfaces, singleton by default /// </summary> /// <param name="services">services</param> /// <param name="typesFilter">filter types to register</param> /// <param name="assemblies">assemblies</param> /// <returns>services</returns> public static IServiceCollection RegisterAssemblyTypesAsImplementedInterfaces(this IServiceCollection services, Func<Type, bool> typesFilter, params Assembly[] assemblies) => RegisterAssemblyTypesAsImplementedInterfaces(services, typesFilter: typesFilter, ServiceLifetime.Singleton, assemblies); /// <summary> /// RegisterTypeAsImplementedInterfaces /// </summary> /// <param name="services">services</param> /// <param name="typesFilter">filter types to register</param> /// <param name="serviceLifetime">service lifetime</param> /// <param name="assemblies">assemblies</param> /// <returns>services</returns> public static IServiceCollection RegisterAssemblyTypesAsImplementedInterfaces(this IServiceCollection services, Func<Type, bool> typesFilter, ServiceLifetime serviceLifetime, params Assembly[] assemblies) { if (assemblies == null || assemblies.Length == 0) { assemblies = ReflectHelper.GetAssemblies(); } var types = assemblies .Select(assembly => assembly.GetExportedTypes()) .SelectMany(t => t); if (typesFilter != null) { types = types.Where(typesFilter); } foreach (var type in types) { foreach (var implementedInterface in type.GetImplementedInterfaces()) { services.Add(new ServiceDescriptor(implementedInterface, type, serviceLifetime)); } } return services; }
注册类型,这至关于 AutoFac
中 RegisterAssemblyTypes().AsSelf()
htm
services.RegisterAssemblyTypes(t => !t.IsAbstract && t.IsClass && t.IsAssignableTo<IEventHandler>(), typeof(NoticeViewEventHandler).Assembly);
实现源码:接口
/// <summary> /// RegisterAssemblyTypes /// </summary> /// <param name="services">services</param> /// <param name="assemblies">assemblies</param> /// <returns>services</returns> public static IServiceCollection RegisterAssemblyTypes(this IServiceCollection services, params Assembly[] assemblies) => RegisterAssemblyTypes(services, null, ServiceLifetime.Singleton, assemblies); /// <summary> /// RegisterAssemblyTypes /// </summary> /// <param name="services">services</param> /// <param name="serviceLifetime">service lifetime</param> /// <param name="assemblies">assemblies</param> /// <returns>services</returns> public static IServiceCollection RegisterAssemblyTypes(this IServiceCollection services, ServiceLifetime serviceLifetime, params Assembly[] assemblies) => RegisterAssemblyTypes(services, null, serviceLifetime, assemblies); /// <summary> /// RegisterAssemblyTypes /// </summary> /// <param name="services">services</param> /// <param name="typesFilter">filter types to register</param> /// <param name="assemblies">assemblies</param> /// <returns>services</returns> public static IServiceCollection RegisterAssemblyTypes(this IServiceCollection services, Func<Type, bool> typesFilter, params Assembly[] assemblies) => RegisterAssemblyTypes(services, typesFilter, ServiceLifetime.Singleton, assemblies); /// <summary> /// RegisterAssemblyTypes /// </summary> /// <param name="services">services</param> /// <param name="typesFilter">filter types to register</param> /// <param name="serviceLifetime">service lifetime</param> /// <param name="assemblies">assemblies</param> /// <returns>services</returns> public static IServiceCollection RegisterAssemblyTypes(this IServiceCollection services, Func<Type, bool> typesFilter, ServiceLifetime serviceLifetime, params Assembly[] assemblies) { if (assemblies == null || assemblies.Length == 0) { assemblies = ReflectHelper.GetAssemblies(); } var types = assemblies .Select(assembly => assembly.GetExportedTypes()) .SelectMany(t => t); if (typesFilter != null) { types = types.Where(typesFilter); } foreach (var type in types) { services.Add(new ServiceDescriptor(type, type, serviceLifetime)); } return services; }
在 AutoFac 中有一个 Module 的概念能够让咱们更方便的注册,通常能够在一个 Module 里注册须要服务注册的服务,这样注册的好处在于容易作插件化,新的插件须要注册什么样的服务,外界并不知道,若是用 Module 的注册方式,我只须要调用插件中的 Module 就能够注册所须要的服务了
相似于 AutoFac 中的 Module,咱们也能够定义一个 IServiceModule
,定义一个 ConfigueServices(IServiceCollection services)
的方法用来注册服务
public interface IServiceModule { void ConfigureServices(IServiceCollection services); }
public class BusinessServiceModule : IServiceModule { public void ConfigureServices(IServiceCollection services) { services.AddEFRepository(); services.AddBLL(); services.TryAddScoped<ReservationHelper>(); services.TryAddSingleton<CaptchaVerifyHelper>(); services.TryAddSingleton<OperLogHelper>(); } }
上面咱们定义了一个 Module,下面来定义一个扩展方法来注册 module 中的服务信息
使用示例:
// load service modules services.RegisterAssemblyModules(pluginAssemblies);
实现源码:
/// <summary> /// RegisterAssemblyModules /// </summary> /// <param name="services">services</param> /// <param name="assemblies">assemblies</param> /// <returns>services</returns> public static IServiceCollection RegisterAssemblyModules( [NotNull] this IServiceCollection services, params Assembly[] assemblies) { if (assemblies == null || assemblies.Length == 0) { assemblies = ReflectHelper.GetAssemblies(); } foreach (var type in assemblies.SelectMany(ass => ass.GetTypes()) .Where(t => t.IsClass && !t.IsAbstract && typeof(IServiceModule).IsAssignableFrom(t)) ) { try { if (Activator.CreateInstance(type) is IServiceModule module) { module.ConfigureServices(services); } } catch (Exception e) { Console.WriteLine(e); } } return services; }