.NET 6 整合 Autofac 依赖注入容器

前言

一行业务代码还没写,框架代码一大堆,不利于学习。
常看到java的学习资料或博客,标题一般为《SpringBoot 整合 XXX》,所以仿照着写了《.NET 6 整合 Autofac 依赖注入容器》这样一个标题。
以下是我自己的用法,可能不是最佳实践。

一. 引用包

NuGet搜索并安装:
Autofac
Autofac.Extensions.DependencyInjection

二. 配置代码

var builder = WebApplication.CreateBuilder(args);

...省略

// ASP.NET Core整合Autofac
builder.Host.UseServiceProviderFactory(new AutofacServiceProviderFactory());//通过工厂替换,把Autofac整合进来
builder.Host.ConfigureContainer<ContainerBuilder>(containerBuilder =>
{
    ServiceFactory.SetBuilder(containerBuilder);
    ServiceFactory.RegisterAssembly(Assembly.GetExecutingAssembly()); //注册服务
});

var app = builder.Build();

ServiceFactory.SetContainer((app.Services as AutofacServiceProvider).LifetimeScope as IContainer);
Task.Run(async () => await ServiceFactory.StartAllService()); //启动服务,注意:服务启动完成之前,调用接口会异常

...省略

三. 服务的自动注册、启动、停止

IService接口

/// <summary>
/// 服务接口
/// </summary>
public interface IService
{
    /// <summary>
    /// 服务启动
    /// </summary>
    Task OnStart();

    /// <summary>
    /// 服务停止
    /// </summary>
    Task OnStop();
}

RegisterServiceAttribute特性

/// <summary>
/// 注册为服务
/// </summary>
public class RegisterServiceAttribute : Attribute
{
}

ServiceBase抽像类

[RegisterService]
public abstract class ServiceBase : IService
{
    #region OnStart
    public virtual Task OnStart()
    {
        return Task.CompletedTask;
    }
    #endregion

    #region OnStop
    public virtual Task OnStop()
    {
        return Task.CompletedTask;
    }
    #endregion

}

ServiceFactory服务工厂类

public class ServiceFactory
{
    private static ContainerBuilder _builder;

    private static IContainer _container;

    private static bool _isRunning; //服务是否正在运行

    public static void SetBuilder(ContainerBuilder builder)
    {
        _builder = builder;
    }

    public static void SetContainer(IContainer container)
    {
        _container = container;
    }

    /// <summary>
    /// Autofac.ContainerBuilder
    /// </summary>
    public static ContainerBuilder Builder
    {
        get
        {
            return _builder;
        }
    }

    /// <summary>
    /// 获取服务
    /// </summary>
    /// <typeparam name="T">接口类型</typeparam>
    public static T Get<T>()
    {
        if (_isRunning)
        {
            return _container.Resolve<T>();
        }
        else
        {
            throw new Exception("服务尚未启动完成");
        }
    }

    /// <summary>
    /// 获取服务
    /// </summary>
    /// <param name="type">接口类型</param>
    public static object Get(Type type)
    {
        if (_isRunning)
        {
            return _container.Resolve(type);
        }
        else
        {
            throw new Exception("服务尚未启动完成");
        }
    }

    #region 注册程序集
    /// <summary>
    /// 注册程序集
    /// </summary>
    /// <param name="serviceAssembly">服务程序集</param>
    public static void RegisterAssembly(Assembly serviceAssembly)
    {
        Type[] typeArr = serviceAssembly.GetTypes();

        foreach (Type type in typeArr)
        {
            if (type.GetCustomAttribute<RegisterServiceAttribute>() != null && !type.IsAbstract)
            {
                _builder.RegisterType(type).SingleInstance().AsImplementedInterfaces().AsSelf();
            }
            else
            {
                Type[] interfaceTypes = type.GetInterfaces();
                foreach (Type interfaceType in interfaceTypes)
                {
                    if (interfaceType.GetCustomAttribute<RegisterServiceAttribute>() != null && !type.IsAbstract)
                    {
                        _builder.RegisterType(type).SingleInstance().AsImplementedInterfaces().AsSelf();
                        break;
                    }
                }
            }
        }

        //_container = _builder.Build();
    }
    #endregion

    #region 启动所有服务
    /// <summary>
    /// 启动所有服务
    /// </summary>
    public static async Task StartAllService()
    {
        Type iServiceInterfaceType = typeof(IService);
        IEnumerable<Type> types = _container.ComponentRegistry.Registrations.Select(a => a.Activator.LimitType);
        await Parallel.ForEachAsync(types, async (type, c) =>
        {
            if (iServiceInterfaceType.IsAssignableFrom(type))
            {
                object obj = _container.Resolve(type);

                try
                {
                    IService service = obj as IService;
                    await service.OnStart();
                    LogUtil.Info("服务 " + obj.GetType().FullName + " 已启动");
                }
                catch (Exception ex)
                {
                    LogUtil.Error(ex, "服务 " + obj.GetType().FullName + " 启动失败");
                }
            }
        });
        _isRunning = true;
    }
    #endregion

    #region 停止所有服务
    /// <summary>
    /// 停止所有服务
    /// </summary>
    public static async Task StopAllService()
    {
        Type iServiceInterfaceType = typeof(IService);
        IEnumerable<Type> types = _container.ComponentRegistry.Registrations.Select(a => a.Activator.LimitType);
        await Parallel.ForEachAsync(types, async (type, c) =>
        {
            if (iServiceInterfaceType.IsAssignableFrom(type))
            {
                object obj = _container.Resolve(type);
                IService service = obj as IService;

                try
                {
                    await service.OnStop();
                    LogUtil.Info("服务 " + obj.GetType().FullName + " 已停止");
                }
                catch (Exception ex)
                {
                    LogUtil.Error(ex, "服务 " + obj.GetType().FullName + " 停止失败");
                }
            }
        });
        _isRunning = false;
    }
    #endregion

}

ServiceFactory的简写类

/// <summary>
/// ServiceFactory的简写
/// </summary>
public class SF
{
    /// <summary>
    /// 获取服务
    /// </summary>
    /// <typeparam name="T">接口类型</typeparam>
    public static T Get<T>()
    {
        return ServiceFactory.Get<T>();
    }

    /// <summary>
    /// 获取服务
    /// </summary>
    /// <param name="type">接口类型</param>
    public static object Get(Type type)
    {
        return ServiceFactory.Get(type);
    }
}

编写服务类

服务类示例1

/// <summary>
/// 测试服务
/// </summary>
public class TestService : ServiceBase
{
    public async Task<string> GetValue()
    {
        return await Task.FromResult("测试数据");
    }
}

服务类示例2

重写了OnStart和OnStop以实现服务的启动和停止。

/// <summary>
/// 定时任务管理
/// </summary>
public class ScheduleJobs : ServiceBase
{
    #region 变量
    private IScheduler _scheduler;
    #endregion

    #region OnStart
    public override async Task OnStart()
    {
        try
        {
            NameValueCollection options = new NameValueCollection();
            string schedulerName = "DefaultQuartzScheduler";
            options.Add("quartz.scheduler.instanceName", schedulerName);
            StdSchedulerFactory schedulerFactory = new StdSchedulerFactory(options);
            _scheduler = await schedulerFactory.GetScheduler(schedulerName);
            if (_scheduler == null)
            {
                _scheduler = await schedulerFactory.GetScheduler();
            }
            await _scheduler.Start();
            AddJobs(_scheduler);
        }
        catch (Exception ex)
        {
            Console.WriteLine(ex.ToString());
        }
    }
    #endregion

    #region OnStop
    public override async Task OnStop()
    {
        await _scheduler.Shutdown();
    }
    #endregion

    #region ScheduleJob
    private async Task ScheduleJob<T>(IScheduler scheduler, string cronString) where T : IJob
    {
        IJobDetail jobDetail = JobBuilder.Create<T>().Build();
        ITrigger trigger = TriggerBuilder.Create().WithCronSchedule(cronString).Build();
        await scheduler.ScheduleJob(jobDetail, trigger);
    }
    #endregion

    private async void AddJobs(IScheduler scheduler)
    {
        await ScheduleJob<TestJob>(scheduler, "0/30 * * * * ?");
    }

}

构造注入示例

[ApiController]
[Route("[controller]")]
public class TestController : ControllerBase
{
    private readonly TestService _testService;

    public TestController(TestService testService)
    {
        _testService = testService;
    }

    ...省略
}

也可以这样使用

var value = await SF.Get<TestService>().GetValue();

原文链接:https://www.cnblogs.com/s0611163/archive/2023/04/26/17356403.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:.NET 6 整合 Autofac 依赖注入容器 - Python技术站

(0)
上一篇 2023年4月27日
下一篇 2023年4月27日

相关文章

  • C#中利用代理实现观察者设计模式详解

    以下是实现“C#中利用代理实现观察者设计模式”的完整攻略: 1. 观察者设计模式 观察者设计模式是一种行为设计模式,允许一个对象(称为主题或可观察对象)在状态改变时自动通知其依赖对象(称为观察者)。 2. 利用代理实现观察者模式 在C#中,我们可以利用代理来实现观察者模式。代理可以用于委托方法,在调用方法之前进行一些操作,这有利于观察者模式的实现。 以下是实…

    C# 2023年6月3日
    00
  • C#给Word中的字符添加着重号的方法详解

    C#给Word中的字符添加着重号的方法详解 在C#中,可以使用Microsoft.Office.Interop.Word来实现对Word文档的自动操作。以下是给Word中的字符添加着重号的方法的详细攻略。 前置条件 安装Microsoft.Office.Interop.Word。 引用Microsoft.Office.Interop.Word。 实现步骤 创…

    C# 2023年6月7日
    00
  • C# char[]与string byte[]与string之间的转换详解

    C# char[]与string 在C#中,char[]与string之间的转换可以通过以下方法实现: char[]转string 可以调用string构造函数,传入char[]即可: char[] chars = { ‘H’, ‘e’, ‘l’, ‘l’, ‘o’ }; string str = new string(chars); 上面的代码会将char…

    C# 2023年6月8日
    00
  • C#判断字符串中是否包含指定字符串及contains与indexof方法效率问题

    C#中判断一个字符串是否包含子字符串是一个常用的任务。本文将讲解如何使用C#的contains和indexof方法来实现这个任务,并探讨它们的效率问题。 contains方法 contains方法是String类中的一种方法,用于判断一个字符串是否包含指定的子字符串。代码示例如下: string str1 = "hello world";…

    C# 2023年6月8日
    00
  • C#中间语言及ILDASM工具用法

    C#编译器会将C#源代码编译成一个叫做CIL(Common Intermediate Language,公用中间语言)的中间语言,这个中间语言是基于栈的虚拟机语言。CIL是一种基于栈的虚拟机语言,每个CIL指令都会入栈一个或多个值,并且返回结果也会通过栈来返回。 ILDASM是一个IL(Intermediate Language,中间语言)反汇编器,允许用户…

    C# 2023年6月3日
    00
  • C#使用BitConverter与BitArray类进行预定义基础类型转换

    下面是“C#使用BitConverter与BitArray类进行预定义基础类型转换”的完整攻略。 什么是BitConverter和BitArray类? 在C#中,我们可以使用BitConverter和BitArray类来进行预定义基础类型的转换。其中,BitConverter类用于在基本数据类型和字节数组之间进行转换,而BitArray类可以对比特位进行操作…

    C# 2023年6月8日
    00
  • C#编程简单实现生成PDF文档的方法示例

    综述 在C#编程中生成PDF文档是一项非常有用的功能。PDF文档可以在任何设备上展示并保留格式,因此它可以被广泛应用于数据报表、电子书等领域。本文将会详细讲述如何使用C#编程实现生成PDF文档的方法,同时提供两个实例来演示如何操作。 步骤 引入PDF库 在C#中生成PDF文档之前,需要下载和使用第三方PDF库。有很多开源的PDF生成库可以方便的使用,例如iT…

    C# 2023年6月1日
    00
  • Winform界面中实现菜单列表的动态个性化配置管理方法

    Winform界面中实现菜单列表的动态个性化配置管理方法,可以通过以下步骤来实现: 1. 数据存储 首先,需要将菜单列表的配置信息存储在数据源中,例如使用XML或JSON格式。在这个数据源中,每个菜单项都有自己的属性包括菜单名称、菜单ID、菜单所处的层级以及是否可见等。 例如,可以使用以下XML格式的数据源来存储菜单列表的信息: <MenuItems&…

    C# 2023年6月7日
    00
合作推广
合作推广
分享本页
返回顶部