Autofac

Posted 彪悍的代码不需要注释

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Autofac相关的知识,希望对你有一定的参考价值。

Autofac是一款IOC框架,比较于其他的IOC框架,如Spring.NET,Unity,Castle等等所包含的,它很轻量级性能上非常高。

官方网站http://autofac.org/

源码下载地址https://github.com/autofac/Autofac

  • 控制反转(IoC/Inverse Of Control):   调用者不再创建被调用者的实例,由autofac框架实现(容器创建)所以称为控制反转。
  • 依赖注入(DI/Dependence injection) :   容器创建好实例后再注入调用者称为依赖注入。

安装Autofac

Install-Package Autofac

1. 扫描类型 RegisterAssemblyTypes(接受1个或者多个程序集的数组),程序集必须是public的

  1.1  过滤类型:Where()

  

builder.RegisterAssemblyTypes(Assembly.LoadFile(servicesDllFile))
       .Where(t => t.Name.EndsWith("Services"));

  

     1.2 排除类型: Except()

  

builder.RegisterAssemblyTypes(Assembly.LoadFile(servicesDllFile))
       .Except<myType>(ct =>ct.As<ISpecial>().SingleInstance());

  

AsImplementedInterfaces() : 表示注册的类型,以接口的方式注册,但不包括IDisposable接口

builder.RegisterAssemblyTypes(Assembly.LoadFile(servicesDllFile))
       .Where(t => t.Name.EndsWith("Repository"))
       .AsImplementedInterfaces();

AsClosedTypesOf(open):可分配给注册类型一个接近开放泛型类型的实例

builder.RegisterAssemblyTypes(Assembly.LoadFile(servicesDllFile))
       .AsClosedTypesOf(typeof(IRepository<>));

AsSelf() :默认: 注册类型本身 - 当重写其他服务默认规范时非常有用

builder.RegisterAssemblyTypes(Assembly.LoadFile(servicesDllFile))
       .AsImplementedInterfaces()
       .AsSelf();

 2. 扫描模块 RegisterAssemblyModules() 册方法执行,它的名字是什么,就执行哪个,它通过Autofac 模块提供的程序集扫描,创建模块实例,然后使用当前container builder 注册他们。使用System.Web.Compilation.BuildManager中的GetReferencedAssemblies()方法,会得到一个引用程序集列表,

var assemblies = BuildManager.GetReferencedAssemblies().Cast<Assembly>();

 


注入方法一:

var builder = new ContainerBuilder();

builder.RegisterType<TestService>();
builder.RegisterType<TestDao>().As<ITestDao>();

return builder.Build();

注入方法二:

为了统一管理 IoC 相关的代码,并避免在底层类库中到处引用 Autofac 这个第三方组件,定义了一个专门用于管理需要依赖注入的接口与实现类的空接口 IDependency:

  /// <summary>
  /// 依赖注入接口,表示该接口的实现类将自动注册到IoC容器中
  /// </summary>
  public interface IDependency
  { }

  

这个接口没有任何方法,不会对系统的业务逻辑造成污染,所有需要进行依赖注入的接口,都要继承这个空接口,例如:

业务单元操作接口:

/// <summary>
/// 业务单元操作接口
/// </summary>
public interface IUnitOfWork : IDependency
{
    ...
}

  Autofac 是支持批量子类注册的,有了 IDependency 这个基接口,我们只需要 Global 中很简单的几行代码,就可以完成整个系统的依赖注入匹配:

ContainerBuilder builder = new ContainerBuilder();
builder.RegisterGeneric(typeof(Repository<,>)).As(typeof(IRepository<,>));
Type baseType = typeof(IDependency);

// 获取所有相关类库的程序集
Assembly[] assemblies = ...

builder.RegisterAssemblyTypes(assemblies)
    .Where(type => baseType.IsAssignableFrom(type) && !type.IsAbstract)
    .AsImplementedInterfaces().InstancePerLifetimeScope();//InstancePerLifetimeScope 保证对象生命周期基于请求
IContainer container = builder.Build();
DependencyResolver.SetResolver(new AutofacDependencyResolver(container));

  如此,只有站点主类库需要引用 Autofac,而不是到处都存在着注入的相关代码,大大降低了系统的复杂度。

1、InstancePerDependency

对每一个依赖或每一次调用创建一个新的唯一的实例。这也是默认的创建实例的方式。

2、InstancePerLifetimeScope

在一个生命周期域中,每一个依赖或调用创建一个单一的共享的实例,且每一个不同的生命周期域,实例是唯一的,不共享的。

3、InstancePerMatchingLifetimeScope

在一个做标识的生命周期域中,每一个依赖或调用创建一个单一的共享的实例。打了标识了的生命周期域中的子标识域中可以共享父级域中的实例。若在整个继承层次中没有找到打标识的生命周期域,则会抛出异常:DependencyResolutionException。

4、InstancePerOwned

在一个生命周期域中所拥有的实例创建的生命周期中,每一个依赖组件或调用Resolve()方法创建一个单一的共享的实例,并且子生命周期域共享父生命周期域中的实例。若在继承层级中没有发现合适的拥有子实例的生命周期域,则抛出异常:DependencyResolutionException。

5、SingleInstance

每一次依赖组件或调用Resolve()方法都会得到一个相同的共享的实例。其实就是单例模式。

6、InstancePerHttpRequest

在一次Http请求上下文中,共享一个组件实例。仅适用于asp.net mvc开发。

参考链接:

autofac 创建实例方法总结:http://www.cnblogs.com/manglu/p/4115128.html

AutoFac使用方法总结:Part I:http://niuyi.github.io/blog/2012/04/06/autofac-by-unit-test/

以上是关于Autofac的主要内容,如果未能解决你的问题,请参考以下文章

autofac v4.0+通过配置文件的方式注册组件

Autofac使用代码方式进行组件注册不需要依赖

浅析依赖注入框架Autofac的使用

用 autofac 注册抽象类

Autofac

Autofac 依赖注入框架