一、为什么使用AutoFac?

之前介绍了Unity和Ninject两个IOC容器,但是发现园子里用AutoFac的貌似更为普遍,于是捯饬了两天,发现这个东东确实是个高大上的IOC容器~

Autofac是.NET领域最为流行的IOC框架之一,传说是速度最快的一个:

优点: 

  • 它是C#语言联系很紧密,也就是说C#里的很多编程方式都可以为Autofac使用,例如可以用Lambda表达式注册组件
  • 较低的学习曲线,学习它非常的简单,只要你理解了IoC和DI的概念以及在何时需要使用它们
  • XML配置支持
  • 自动装配
  • 与Asp.Net MVC 3集成
  • 微软的Orchad开源程序使用的就是Autofac,从该源码可以看出它的方便和强大

既然它都这么牛X了,我们用它就理所当然了,所以推荐其为IOC的终极解决方案!

二、AutoFac的使用

首先你必须获取AutoFac,这里你可以通过各种方式加载它,我这里还是通过VS中的NuGet来加载AutoFac,不论是哪种方式,最终的目的就是将 Autofac.dll,Autofac.Configuration.dll 这两个程序集引用到你的项目中。这样在你的项目中,如果想使用AutoFac,只需添加其命名空间引用即可~

1、AutoFac入门

我们先定义一个数据访问的接口:

1. public
2. {  
3. void
4. }


然后用Sql和Oracle两种方式分别实现上述接口,不过这里只是演示而已,所以并没有真正去实现这两个类,你懂的~

SQL方式:


1. public class
2. {  
3. public void
4.     {  
5. "使用sqlDAL添加相关信息");  
6.     }  
7. }


Oracle方式:

1. public class
2. {  
3. public void
4.     {  
5. "使用OracleDAL添加相关信息");  
6.     }  
7. }


然后注入实现构造函数注入:

1. public class
2. {   
3.     IDAL _dal;  
4. public
5.     {   
6.         _dal= dal;  
7.     }  
8. public void
9.     {   
10.         _dal.Insert(commandText);   
11.     }  
12. }

最后要真正完成依赖注入就得AtuoFac登场了:


1. var builder = new
2. builder.RegisterType<DBManager>();   
3. builder.RegisterType<SqlDAL>().As<IDAL>();   
4. using
5. {   
6.     var manager = container.Resolve<DBManager>();   
7. "INSERT INTO Persons VALUES ('Man', '25', 'WangW', 'Shanghai')");   
8. }


从以上栗子可以看出,其实AutoFac的使用跟Unity的使用有点像,关键的东东就是这个Container容器类

2、AutoFac常用方法说明

(1)builder.RegisterType<Object>().As<Iobject>():注册类型及其实例。例如下面就是注册接口IDAL的实例SqlDAL


    1. ContainerBuilder builder = new
    2. builder.RegisterType<SqlDAL>().As<IDAL>();  
    3. IContainer container = builder.Build();  
    4. SqlDAL sqlDAL = (SqlDAL)container.Resolve<IDAL>();


    (2)IContainer.Resolve<IDAL>():解析某个接口的实例。例如上面的最后一行代码就是解析IDAL的实例SqlDAL

    (3)builder.RegisterType<Object>().Named<Iobject>(string name):为一个接口注册不同的实例。有时候难免会碰到多个类映射同一个接口,比如SqlDAL和OracleDAL都实现了IDAL接口,为了准确获取想要的类型,就必须在注册时起名字。


      1. builder.RegisterType<SqlDAL>().Named<IDAL>("sql");  
      2. builder.RegisterType<OracleDAL>().Named<IDAL>("oracle");  
      3. IContainer container = builder.Build();  
      4. SqlDAL sqlDAL = (SqlDAL)container.ResolveNamed<IDAL>("sql");  
      5. OracleDAL oracleDAL = (OracleDAL)container.ResolveNamed<IDAL>("oracle");


      (4)IContainer.ResolveNamed<IDAL>(string name):解析某个接口的“命名实例”。例如上面的最后一行代码就是解析IDAL的命名实例OracleDAL

      (5)builder.RegisterType<Object>().Keyed<Iobject>(Enum enum):以枚举的方式为一个接口注册不同的实例。有时候我们会将某一个接口的不同实现用枚举来区分,而不是字符串,例如:


      1. public enum


        1. builder.RegisterType<SqlDAL>().Keyed<IDAL>(DBType.Sql);  
        2. builder.RegisterType<OracleDAL>().Keyed<IDAL>(DBType.Oracle);  
        3. IContainer container = builder.Build();  
        4. SqlDAL sqlDAL = (SqlDAL)container.ResolveKeyed<IDAL>(DBType.Sql);  
        5. OracleDAL oracleDAL = (OracleDAL)container.ResolveKeyed<IDAL>(DBType.Oracle);


        (6)IContainer.ResolveKeyed<IDAL>(Enum enum):根据枚举值解析某个接口的特定实例。例如上面的最后一行代码就是解析IDAL的特定实例OracleDAL

        (7)builder.RegisterType<Worker>().InstancePerDependency():用于控制对象的生命周期,每次加载实例时都是新建一个实例,默认就是这种方式

        (8)builder.RegisterType<Worker>().SingleInstance():用于控制对象的生命周期,每次加载实例时都是返回同一个实例

        (9)IContainer.Resolve<T>(NamedParameter namedParameter):在解析实例T时给其赋值


        1. DBManager manager = container.Resolve<DBManager>(new NamedParameter("name", "SQL"));


          1. public class
          2. {     
          3.     IDAL dal;  
          4. public
          5.     {  
          6.         Name = name;  
          7.         dal= _dal;  
          8.     }  
          9. }


          3、通过配置的方式使用AutoFac

          (1)先配置好配置文件

          1. <?xml version="1.0"?>
          2. <configuration> 
          3.   <configSections> 
          4.     <section name="autofac" type="Autofac.Configuration.SectionHandler, Autofac.Configuration"/>
          5.   </configSections> 
          6.   <autofac defaultAssembly="ConsoleApplication1">
          7.     <components> 
          8.       <component type="ConsoleApplication1.SqlDAL, ConsoleApplication1" service="ConsoleApplication1.IDAL" />
          9.     </components> 
          10.   </autofac> 
          11. </configuration>


          (2)读取配置实现依赖注入(注意引入Autofac.Configuration.dll)


          1. static void
          2.     {  
          3. new
          4.         builder.RegisterType<DBManager>();  
          5. new ConfigurationSettingsReader("autofac"));  
          6. using
          7.         {  
          8.             DBManager manager = container.Resolve<DBManager>();  
          9. "INSERT INTO Persons VALUES ('Man', '25', 'WangW', 'Shanghai')");   
          10.     }