.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日

相关文章

  • LINQ投影操作符Select与限制操作符where介绍

    LINQ是.NET Framework中的一种强大的查询语言,能够对各种数据集合进行查询操作。LINQ提供了多个操作符,其中投影操作符Select和限制操作符where是常用的两种操作符。 Select操作符 Select操作符是一种投影操作符,用于通过从集合中选择特定的项创建新集合。Select操作符的语法如下: IEnumerable<TResul…

    C# 2023年6月1日
    00
  • .NET Core配置多环境的方法步骤

    .NET Core 配置多环境的方法步骤 在 .NET Core 中,我们可以使用多环境配置来管理不同环境下的应用程序配置。本攻略将介绍如何在 .NET Core 中配置多环境。 步骤 以下是在 .NET Core 中配置多环境的步骤: 创建 appsettings.json 文件。 在项目根目录下创建 appsettings.json 文件,并添加以下内容…

    C# 2023年5月17日
    00
  • C#多线程的相关操作讲解

    C#多线程的相关操作讲解 在 C# 中,可以通过多线程机制来使一个程序同时执行多个任务,更好地利用计算资源,提高程序的效率和性能。本篇文章将针对 C# 多线程相关操作进行详细讲解,内容包括线程的创建、启动、停止,线程同步和互斥,以及线程池等多方面。 一、线程的创建和启动 C# 中可以使用 Thread 类来创建和启动线程。Thread 构造函数有两个重载形式…

    C# 2023年5月15日
    00
  • c#判断字符是否为中文的三种方法分享(正则表达式判断)

    当我们需要实现c#中判断一个字符是否为中文时,可以运用以下三种方法进行判断: 1. Unicode码判断法 Unicode码代表着一个全球通用的编码标准,它为每个字符分配了一个唯一的标识。 中文的Unicode编码范围为 4E00 ~ 9FFF,因此可以通过以下代码实现中文判断: public static bool IsChinese(char c) { …

    C# 2023年6月8日
    00
  • asp.net连接查询SQL数据库并把结果显示在网页上(2种方法)

    下面是详细讲解“asp.net连接查询SQL数据库并把结果显示在网页上(2种方法)”的完整攻略。 1. 简介 ASP.NET是Microsoft开发的一种Web应用程序开发技术,它使用.NET框架,支持多种编程语言,如C#、VB等。在ASP.NET中,我们可以使用ADO.NET连接到SQL Server数据库并执行查询操作,然后把结果显示在网页上。 2. 方…

    C# 2023年5月31日
    00
  • C#中Activator.CreateInstance()方法用法分析

    C#中Activator.CreateInstance()方法用法分析 前言 C#中 Activator.CreateInstance() 方法是一种动态创建实例的方法。在实际项目中,我们常常需要动态加载程序集并创建其中的类型对象。这时,Activator.CreateInstance()方法就可以起到很好的作用。 方法说明 Activator.Create…

    C# 2023年5月31日
    00
  • C# GDI+实现时钟表盘

    C# GDI+实现时钟表盘的攻略如下: 1. 准备工作 首先需要在项目中引入System.Drawing和System.Drawing.Drawing2D命名空间,然后在代码中创建一个PictureBox控件,这个控件将用来显示时钟。 2. 绘制表盘 我们可以先创建一个空白的位图对象,然后在该对象中绘制表盘的外圆、刻度以及数字等元素。这个过程中需要使用到Gr…

    C# 2023年6月1日
    00
  • C#实现简单的飞行棋小游戏

    C#实现简单的飞行棋小游戏攻略 简介 飞行棋是一款非常有趣的棋类游戏,它的规则简单,易于操作,适合各个年龄段的玩家进行娱乐。本文将使用 C# 语言来实现一个简单的飞行棋小游戏,该游戏支持人人对战和人机对战两种模式,玩家可根据自己的性格和需要来选择不同的模式。本文的攻略适合那些对 C# 语言有一定了解和掌握的读者来学习和参考。 实现思路 飞行棋的实现主要有以下…

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