从0到1搭建权限管理系统系列四 .net8 中Autofac的使用(附源码)

说明

    该文章是属于OverallAuth2.0系列文章,每周更新一篇该系列文章(从0到1完成系统开发)。

    该系统文章,我会尽量说的非常详细,做到不管新手、老手都能看懂。

    说明:OverallAuth2.0 是一个简单、易懂、功能强大的权限+可视化流程管理系统。

友情提醒:本篇文章是属于系列文章,看该文章前,建议先看之前文章,可以更好理解项目结构。

有兴趣的朋友,请关注我吧(*^▽^*)。

关注我,学不会你来打我

关于Autofac

虽然微软有自带的Ioc框架,但是我想说的是Autofac应该说是我用过的最好的Ioc框架。

它的高性能、灵活性、对Aop的支持,简直就是开发人员的梦中情人。

autofac,它能有效的降低繁杂系统中,代码的耦合度,提高代码的可维护性。

不仅如此,它可以轻松替换依赖关系,从而提高测试效率。

安装Autofac

1、在启动项目中,安装最新版Autofac和Autofac.Extensions.DependencyInjection。

创建类库

1、DomainService:领域服务类

2、Infrastructure:基础设施类

3、CoreDomain:核心领域类

如果你看了该系列每篇文章,你会发现,我们的底层架构,是使用DDD领域驱动模型的设计模式搭建后端。

系统架构如图所示:

搭建好系统架构,我们开始使用Autofac

创建AutofacPlugIn类文件

在系统PlugIn文件中创建AutofacPlugIn类文件,内容如下:

/// <summary>
/// Autofac插件
/// </summary>
public class AutofacPlugIn : Autofac.Module
{
    /// <summary>
    /// 重写Autofac的Load方法
    /// </summary>
    /// <param name="containerBuilder"></param>
    protected override void Load(ContainerBuilder containerBuilder)
    {
        //服务项目程序集
        Assembly service = Assembly.Load("DomainService");
        Assembly intracface = Assembly.Load("Infrastructure");

        //项目必须以xxx结尾
        containerBuilder.RegisterAssemblyTypes(service).Where(n => n.Name.EndsWith("Service") && !n.IsAbstract)
            .InstancePerLifetimeScope().AsImplementedInterfaces();
        containerBuilder.RegisterAssemblyTypes(intracface).Where(n => n.Name.EndsWith("Repository") && !n.IsAbstract)
           .InstancePerLifetimeScope().AsImplementedInterfaces();

    }

}

代码解释:

获取DomainService和Infrastructure程序集,然后批量注入。

值得注意的是,我们在创建类时,给了限定条件。那就是程序集下面的类,必须以Service和Repository结尾,否则autofac会抛出错误。

当然有以xxx结尾,那必然有以xxx开头。使用StartsWith轻松解决。

在Program组成Autofac

.net8 autofac注册和以往有点出去,需要在Program中添加如如下代码

var builder = WebApplication.CreateBuilder(args);
//自定义Autofac中间件
builder.Host.UseServiceProviderFactory(new AutofacServiceProviderFactory()).ConfigureContainer<ContainerBuilder>(builder =>
 {
     builder.RegisterModule<AutofacPlugIn>();
 });

经过以上配置,autofac已经配置好,是不是很简单,接下来测试一下。

使用

在Infrastructure类中建立ISysUserRepository和SysUserRepository仓储接口及接口实现

结构如下:

内容代码如下:

ISysUserRepository

/// <summary>
/// 用户服务仓储接口
/// </summary>
public interface ISysUserRepository
{
    /// <summary>
    /// 测试Autofac
    /// </summary>
    /// <returns></returns>
    string TestAutofac();
}

SysUserRepository

  /// <summary>
  /// 用户服务仓储接口实现
  /// </summary>
  public class SysUserRepository : ISysUserRepository
  {
      /// <summary>
      /// 测试Autofac
      /// </summary>
      /// <returns></returns>
      public string TestAutofac()
      {
          return "Autofac使用成功";
      }
  }

在DomainService中创建ISysUserService和SysUserService服务接口和实现

结构如下:

内容代码如下

ISysUserService

/// <summary>
/// 用户服务接口
/// </summary>
public interface ISysUserService
{
    /// <summary>
    /// 测试Autofac
    /// </summary>
    /// <returns></returns>
    string TestAutofac();
}

SysUserService

/// <summary>
/// 用户服务接口实现
/// </summary>
public class SysUserService : ISysUserService
{
    #region 构造实例化

    private readonly ISysUserRepository _sysUserRepository;

    public SysUserService(ISysUserRepository sysUserRepository)
    {
        _sysUserRepository = sysUserRepository;
    }

    #endregion

    /// <summary>
    /// 测试Autofac
    /// </summary>
    /// <returns></returns>
    public string TestAutofac()
    {
        return _sysUserRepository.TestAutofac();
    }
}

测试

在SysUserController控制器中调用接口(该控制器,在之前文章中已经建好,没看之前文章的,随意见一个webapi的控制器也行)。

代码如下:

 #region 构造实列化

 /// <summary>
 /// 用户服务
 /// </summary>
 public ISysUserService _userService;

 /// <summary>
 /// 构造函数
 /// </summary>
 /// <param name="userService"></param>
 public SysUserController(ISysUserService userService)
 {
     _userService = userService;
 }

 #endregion

 /// <summary>
 /// 获取Token
 /// </summary>
 /// <param name="userName">用户名</param>
 /// <param name="password">密码</param>
 [HttpGet]
 [AllowAnonymous]
 public string GetToken(string userName, string password)
 {
     var loginResult = JwtPlugIn.BuildToken(new LoginInput { UserName = userName, Password = password });

     return loginResult.Token ?? string.Empty;
 }

 /// <summary>
 /// 测试Autofac
 /// </summary>
 /// <returns></returns>
 [HttpGet]
 [AllowAnonymous]
 public string TestAutofac() 
 {
   return  _userService.TestAutofac();
 }

上述代码中要先通过Autofac来实例化接口。

打开swagger,开始测试

之前说道,我们的接口必须以xxx结束,那么现在我们把这个约定改一下,看能否调用成功。

修改代码如下:

 /// <summary>
 /// 重写Autofac的Load方法
 /// </summary>
 /// <param name="containerBuilder"></param>
 protected override void Load(ContainerBuilder containerBuilder)
 {
     //服务项目程序集
     Assembly service = Assembly.Load("DomainService");
     Assembly intracface = Assembly.Load("Infrastructure");

     //项目必须以xxx结尾
     containerBuilder.RegisterAssemblyTypes(service).Where(n => n.Name.EndsWith("Service111111111") && !n.IsAbstract)
         .InstancePerLifetimeScope().AsImplementedInterfaces();
     containerBuilder.RegisterAssemblyTypes(intracface).Where(n => n.Name.EndsWith("Repository2222222") && !n.IsAbstract)
        .InstancePerLifetimeScope().AsImplementedInterfaces();

 }

可以看到,我们把以Service、Repository结束改成了Service111111111、Repository2222222,也就是说,我们的领域服务类和基础仓储类是不符合改命名规范的,aufofac应该报出错误。

测试:

以上就是aufofac在本期内容中的使用。

在接下来的系列中,会使用autofac+aop的结合使用,有兴趣的朋友,请博客园关注、微信关注,感谢观看。

源代码地址:https://gitee.com/yangguangchenjie/overall-auth2.0-web-api  

预览地址:http://139.155.137.144:8880/swagger/index.html

帮我Star,谢谢。

有兴趣的朋友,请关注我微信公众号吧(*^▽^*)。

关注我:一个全栈多端的宝藏博主,定时分享技术文章,不定时分享开源项目。关注我,带你认识不一样的程序世界

posted @ 2024-09-29 16:42  陈逸子风  阅读(1392)  评论(4编辑  收藏  举报