写在开头
ABP开源项目最近有点小火,还开展了线下活动。本着学习DDD的心态与学习开源代码的的好奇,我也看了一遍ABP源码,在此将自己学习ABP的一些心得记录下来。
作为核心的IoC
作为一种解耦的方式,一些IoC框架就成了项目了核心。比如蒋金楠的VideoMall与陈青阳的Byteart Retail项目都是使用Unity。而ABP则是使用Castle。
Register与Resolve
Register与Reslove是IoC基本功能,向容器中注类型对应关系再向容器按注入规则索取对象实体。ABP为Register与Resolve分别定义接口:IIocRegistrar、IocResolver,再由IIocManager继承这两个接口,所以在IIocManager中统一了Register与Resolve,其中IIocManger中定义了IWindsorContainer。再由IocManager去实现IIoCManager.这样就形成了一个完整的, Register与Resolve。类结构如下:
其中,IoCManager当中用了单例去维持Register内容。
生命周期
ABP对于生命周期管理提供了两种方式。
-
手工注入
系统提供了一生命周期的枚举类型:DependencyLifeStyle
/// <summary> /// Lifestyles of types used in dependency injection system. /// </summary> public enum DependencyLifeStyle { /// <summary> /// Singleton object. Created a single object on first resolving /// and same instance is used for subsequent resolves. /// </summary> Singleton, /// <summary> /// Transient object. Created one object for every resolving. /// </summary> Transient }
再IIoCResgister接口中重载了四个方法Register都有DependencyLifeStyle参数
-
系统自动识别
系统识别的方式主要就对整个程序集进行注入,ABP提供了两个有关生命周期的接口,分别是:ISingletonDependency与ITransientDependency
同时IIoCRegister接口中有两个RegisterAssemblyByConvention方法,
这两个方法都用到了Assembly作为参数。
关于这种方式的注入ABP提供了一个Assembly容器接口:IConventionalRegistrationContext与其默认实现ConventionalRegistrationContext,一个Assembly解析接口:IConventionalDependencyRegistrar及其默认实现BasicConventionalRegistrar,还有一配置ContextConventionalRegistrationConfig
这两个接口与一个配置类共同解析程序集中可依赖注入的类型。
其中主要解析与注入工作由IConventionalDependencyRegistrar中的RegisterAssembly方法完成,其具体实现为:
public void RegisterAssembly(IConventionalRegistrationContext context) { //Transient context.IocManager.IocContainer.Register( Classes.FromAssembly(context.Assembly) .IncludeNonPublicTypes() .BasedOn<ITransientDependency>() .WithService.Self() .WithService.DefaultInterfaces() .LifestyleTransient() ); //Singleton context.IocManager.IocContainer.Register( Classes.FromAssembly(context.Assembly) .IncludeNonPublicTypes() .BasedOn<ISingletonDependency>() .WithService.Self() .WithService.DefaultInterfaces() .LifestyleSingleton() ); //Windsor Interceptors context.IocManager.IocContainer.Register( Classes.FromAssembly(context.Assembly) .IncludeNonPublicTypes() .BasedOn<IInterceptor>() .WithService.Self() .LifestyleTransient() ); }
在源代码当中我们可以看到对拦截器(IInterceptor)也有处理
ABP默认只提供两种生命周期类型:Singleton(单例),Transient(实例)两种,对于其他的类型暂时不知道怎么处理。
Resolve的Wrapper
ABP为Resolve提供一个Wrapper,这个Wrapper的主要是对容器对象的Dispose管理
在其实现DisposableDependencyObjectWrapper<T>中有体现:
internal class DisposableDependencyObjectWrapper<T> : IDisposableDependencyObjectWrapper<T> { private readonly IIocResolver _iocResolver; public T Object { get; private set; } public DisposableDependencyObjectWrapper(IIocResolver iocResolver, T obj) { _iocResolver = iocResolver; Object = obj; } public void Dispose() { _iocResolver.Release(Object); } }
AbpCoreInstaller
AbpCoreInstaller主要实现了ABP自身的注入。