美文网首页
Autofac 依赖注入框架 使用

Autofac 依赖注入框架 使用

作者: RobinJiang | 来源:发表于2016-03-04 14:47 被阅读897次

    简介

    Autofac是一款IOC框架,比较于其他的IOC框架,如Spring.NET,Unity,Castle等等所包含的,它很轻量级性能上非常高。
    官方网站http://autofac.org/
    源码下载地址https://github.com/autofac/Autofac
    最新版本下载可以看到,包括源码,示例文档,与之相关的测试项目,生成的DLL文件,其他文档

    控制反转和依赖注入

    关于控制反转和依赖注入的文章和书籍很多,对其定义也解释的也仁者见仁,这里就不赘述了,这是本人(只代表个人观点)的理解:

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

    基本使用

    安装Autofac

    Install-Package Autofac
    

    官方使用简单介绍

    Adding Components

    var builder = new ContainerBuilder();
    

    Autofac can use a Linq expression, a .NET type, or a pre-built instance as a component:

    builder.Register(c => new TaskController(c.Resolve<ITaskRepository>()));
    builder.RegisterType<TaskController>();
    builder.RegisterInstance(new TaskController());
    

    Or, Autofac can find and register the component types in an assembly:

    builder.RegisterAssemblyTypes(controllerAssembly);
    

    Calling Build() creates a container:

    var container = builder.Build();
    

    To retrieve a component instance from a container, a service is requested. By default, components provide their concrete type as a service:

    var taskController = container.Resolve<TaskController>();
    

    To specify that the component’s service is an interface, the As()
    method is used at registration time:

    builder.RegisterType<TaskController>().As<IController>();
    // enabling
    var taskController = container.Resolve<IController>();
    

    方法一:

    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,而不是到处都存在着注入的相关代码,大大降低了系统的复杂度。
    参考:http://www.cnblogs.com/guomingfeng/p/osharp-layer.html


    创建实例方法
    1、InstancePerDependency
    对每一个依赖或每一次调用创建一个新的唯一的实例。这也是默认的创建实例的方式。
    官方文档解释:

    Configure the component so that every dependent component or call to Resolve() gets a new, unique instance (default.)

    2、InstancePerLifetimeScope
    在一个生命周期域中,每一个依赖或调用创建一个单一的共享的实例,且每一个不同的生命周期域,实例是唯一的,不共享的。
    官方文档解释:

    Configure the component so that every dependent component or call to Resolve() within a single ILifetimeScope gets the same, shared instance. Dependent components in different lifetime scopes will get different instances.

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

    Configure the component so that every dependent component or call to Resolve() within a ILifetimeScope tagged with any of the provided tags value gets the same, shared instance. Dependent components in lifetime scopes that are children of the tagged scope will share the parent's instance. If no appropriately tagged scope can be found in the hierarchy an DependencyResolutionException is thrown.

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

    Configure the component so that every dependent component or call to Resolve() within a ILifetimeScope created by an owned instance gets the same, shared instance. Dependent components in lifetime scopes that are children of the owned instance scope will share the parent's instance. If no appropriate owned instance scope can be found in the hierarchy an DependencyResolutionException is thrown.

    5、SingleInstance
    每一次依赖组件或调用Resolve()方法都会得到一个相同的共享的实例。其实就是单例模式。
    官方文档解释:

    Configure the component so that every dependent component or call to Resolve() gets the same, shared instance.

    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 依赖注入框架 使用

          本文链接:https://www.haomeiwen.com/subject/dgyokttx.html