NetCore自带的IOC依赖注入如何实现一个接口多个实现类的注入

提示:一般情况下我们都是使用一个接口一个实现类,但是有一些情况,我们为了实现多态,我们会定义一个接口,多个实现类。这种情况我们在NetCore自带的依赖注入容器中,我们应该怎么来实现呢?

目录

前言

一、什么是依赖注入?

二、使用步骤

1.首先我们写一个扩展服务来批量注入我们的服务

2.默认实现服务TenantServiceBase

3.编写用户扩展服务

4.服务的使用

总结



前言

一般情况下我们都是使用一个接口一个实现类,但是有一些情况,我们为了实现多态,我们会定义一个接口,多个实现类。这种情况我们在NetCore自带的依赖注入容器中,我们应该怎么来实现呢?

一、什么是依赖注入?

依赖注入(Dependency Injection),是这样一个过程:由于某客户类只依赖于服务类的一个接口,而不依赖于具体服务类,所以客户类只定义一个注入点。在程序运行过程中,客户类不直接实例化具体服务类实例,而是客户类的运行上下文环境或专门组件负责实例化服务类,然后将其注入到客户类中,保证客户类的正常运行。

二、使用步骤

1.首先我们写一个扩展服务来批量注入我们的服务

代码如下(示例):

using JuCheap.WebApi.Common;
using JuCheap.WebApi.TenantCustomServices;
using JuCheap.WebApi.TenantFactory;
using JuCheap.WebApi.TenantServices;
using JuCheap.WebApi.Utils;
using Microsoft.Extensions.DependencyInjection;
using System.Linq;
using System.Reflection;

namespace JuCheap.WebApi
{
    /// <summary>
    /// 模块初始化
    /// </summary>
    public static class JuCheapServiceRegistor
    {
        public static void AddJuCheapService(this IServiceCollection services)
        {
            //基础服务注册
            services.AddScoped<ITenantFactoryService, TenantFactoryService>();
            services.AddScoped<IRedisService, RedisService>();
            services.AddScoped<ITenantServiceBase, TenantServiceBase>();

            //批量注入租户级自定义服务
            var serviceRegistrations = typeof(TenantServiceBase).Assembly.GetTypes()
                        .Where(type => type.Namespace != null
                                && type.Namespace.StartsWith("JuCheap.WebApi.TenantCustomServices")
                                && type.GetCustomAttributes<TenantAttribute>().Any(x => x.TenantIds != null && x.TenantIds.Any()))
                        .Select(type => new { Implementation = type })
                        .ToList();
            foreach (var service in serviceRegistrations)
            {
                services.AddScoped(service.Implementation);
            }
        }
    }
}

2.默认实现服务TenantServiceBase

代码如下(示例):

using JuCheap.Common;
using JuCheap.Models;
using JuCheap.TenantServices;
using JuCheap.Utils;

namespace JuCheap.TenantCustomServices
{
    /// <summary>
    /// 租户默认服务
    /// </summary>
    public class TenantServiceBase : ITenantServiceBase
    {
        public virtual string GetDefaultParam(int tenantId)
        {
            return $"{tenantId}-test";
        }
    }

    /// <summary>
    /// 租户默认服务接口
    /// </summary>
    public interface ITenantServiceBase
    {
        string GetDefaultParam(int tenantId);
    }
}


默认服务实现了一个获取默认参数的接口,但是我们不同的用户,获取的参数有可能不一样,有可能做过定制开发等等。

3.编写用户扩展服务

using JuCheap.Common;
using JuCheap.Models;
using JuCheap.TenantServices;
using JuCheap.Utils;

namespace JuCheap.TenantCustomServices
{
    /// <summary>
    /// 用户扩展服务
    /// </summary>
    [Tenant(123456)]
    public class TenantService123456 : TenantServiceBase
    {
        public override string GetDefaultParam(int tenantId)
        {
            return $"{tenantId}-custom-123456";
        }
    }
}

4.服务的使用

默认服务,我们可以通过直接注入ITenantServiceBase接口,就可以使用了,但是我们做过用户扩展服务的,应该怎么使用?我们需要一个工厂的服务接口,如下:

using JuCheap.Common;
using JuCheap.TenantServices;
using Microsoft.Extensions.DependencyInjection;
using System;
using System.Linq;
using System.Reflection;

namespace JuCheap.TenantFactory
{
    /// <summary>
    /// 租户服务获取工厂构造器
    /// </summary>
    public interface ITenantFactoryService
    {
        /// <summary>
        /// 获取租户服务,如果没有自定义服务,则返回默认的TenantBaseService服务
        /// </summary>
        /// <param name="tenantId">租户Id</param>
        /// <returns></returns>
        ITenantServiceBase GetTenantService(int tenantId);
    }

    /// <summary>
    /// 用户服务获取工厂构造器
    /// </summary>
    public class TenantFactoryService : ITenantFactoryService
    {
        private readonly IServiceProvider _serviceProvider;

        public TenantFactoryService(IServiceProvider serviceProvider)
        {
            _serviceProvider = serviceProvider;
        }

        /// <summary>
        /// 获取租户服务,如果没有自定义服务,则返回默认的TenantBaseService服务
        /// </summary>
        /// <param name="tenantId">租户Id</param>
        /// <returns></returns>
        public ITenantServiceBase GetTenantService(int tenantId)
        {
            var serviceRegistrations = typeof(TenantServiceBase).Assembly.GetTypes()
                        .Where(type => type.Namespace != null
                                && type.Namespace.StartsWith("JuCheap.TenantCustomServices")
                                && type.GetCustomAttributes<TenantAttribute>().Any(x => x.TenantIds != null && x.TenantIds.Contains(tenantId)))
                        .ToList();
            if (serviceRegistrations.Any())
            {
                if (serviceRegistrations.Count > 1)
                {
                    throw new Exception($"{tenantId}的租户扩展服务找到多个实现类,请修改");
                }
                return _serviceProvider.GetRequiredService(serviceRegistrations.First()) as ITenantServiceBase;
            }
            //没有找到直接返回默认租户服务
            return _serviceProvider.GetRequiredService<ITenantServiceBase>();
        }
    }
}

当我们的用户有扩展服务的是否,我们使用工厂服务接口来获取扩展服务,如下:

using JuCheap.Models;
using JuCheap.TenantFactory;
using Microsoft.AspNetCore.Mvc;

namespace JuCheap.Controllers
{
    /// <summary>
    /// 账号绑定Api
    /// </summary>
    [Route("api/[controller]/[action]")]
    [ApiController]
    public class BindAccountController : ControllerBase
    {
        private readonly ITenantFactoryService _tenantFactoryService;

        public BindAccountController(ITenantFactoryService tenantFactoryService)
        {
            _tenantFactoryService = tenantFactoryService;
        }

        /// <summary>
        /// 执行绑定验证
        /// </summary>
        [HttpPost]
        public IActionResult Bind([FromBody] BindRequestDTO requestDTO)
        {
            var tenantService = _tenantFactoryService.GetTenantService(User.TenantId);
            tenantService.BindAccount(requestDTO);
            return Ok(true);
        }
    }
}


总结

netcore已经为我们提供了丰富多样的服务注入方式,类似上面的注入方式,我们也可以提供一个Func<int, ITenantServiceBase>的函数来实现,欢迎大家一起讨论。

  • 3
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
NetCore依赖注入是指.NET Core框架提供的一种轻量级和高效的依赖注入机制。依赖注入是一种设计模式,用于实现对象之间的解耦和松耦合。在.Net Core中,依赖注入主要是通过IServiceCollection和IServiceProvider两个核心实现的。 首先,我们需要在Startup.cs文件的ConfigureServices方法中注册依赖关系。通过调用IServiceCollection的AddTransient、AddScoped或AddSingleton方法,我们可以将接口和具体实现型进行关联。这些方法分别表示瞬态注入、作用域注入和单例注入。 然后,在应用程序的其他地方,我们可以通过构造函数参数、属性或方法参数的方式,将依赖对象注入到需要的地方。这些对象会由.Net Core框架自动创建和管理。 依赖注入的好处是,它可以减少代码的耦合性,使得代码更加灵活和可维护。通过将依赖对象的创建和管理交给框架,我们可以专注于业务逻辑的实现,而不需要关心对象的创建细节。同时,依赖注入也方便了单元测试的编写,我们可以轻松地替换依赖对象,进行测试和验证。 另外,依赖注入还可以提高代码的可测试性和可扩展性。通过面向接口编程,我们可以很容易地替换具体实现实现代码的扩展和重用。 总而言之,NetCore依赖注入是一种非常有用的机制,它可以帮助我们开发更加灵活、可维护和可测试的应用程序。它提供了一种简单而强大的方式来管理和解耦各个对象之间的依赖关系。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值