ABP vNext 自动注入,暗藏天坑如斯

导言

我们在使用ABP vNext框架时,都知道该框架为我们实现了自动依赖注入(实现自动注入需要在项目里面创建Module类,并且将Module类上的DependsOn到相应的启动Module类或调用Module类,这步很关键)

自动注入示例

只要我们实现如下接口:
ITransientDependency 瞬时生命周期
ISingletonDependency 全局单例生命周期
IScopedDependency 作用域生命周期

public class Test : ITransientDependency
{
      //todo:....
}

天坑如斯

下面讲一下巨坑,那就是在自动注入的时候系统会自动推断要注入的接口,并且是你实现的所有的接口他都会注入也不是放在第一个实现的接口才会注入,而是根据接口的名字来判断的,比如下面的类就会以接口IUserRepository注入:

public class UserRepository: IUserRepository, ITransientDependency
{
}

同理下面的类型也会以接口IUserRepository注入:

public class EfCoreUserRepository: IUserRepository, ITransientDependency
{
}

但是下面这种情况就不能了

public class UserEfCoreRepository: IUserRepository, ITransientDependency
{
}

源码

Volo.Abp.DependencyInjection.ExposeServicesAttribute.cs

private static List<Type> GetDefaultServices(Type type)
{
    var serviceTypes = new List<Type>();

    foreach (var interfaceType in type.GetTypeInfo().GetInterfaces())
    {
        var interfaceName = interfaceType.Name;

        if (interfaceName.StartsWith("I"))
        {
            interfaceName = interfaceName.Right(interfaceName.Length - 1);
        }

        if (type.Name.EndsWith(interfaceName))
        {
             serviceTypes.Add(interfaceType);
        }
     }
     return serviceTypes;
}

结论

能看出,推断方法是去掉接口名的“I”之后看这个类是否是这个字符串结尾,如果是则注入,如果不是则不注入,并且将会递归他所有实现的接口,如果有多个接口都满足条件,则将会以这些接口进行注入。
还有一点就是实现了这三个自动依赖注入的接口的类型的,都将会以自身类型注入到容器

Tags: