1、爲何使用AutoFac?
以前介紹了Unity和Ninject兩個IOC容器,可是發現園子裏用AutoFac的貌似更爲廣泛,因而捯飭了兩天,發現這個東東確實是個高大上的IOC容器~html
Autofac是.NET領域最爲流行的IOC框架之一,傳說是速度最快的一個:git
優勢: github
- 它是C#語言聯繫很緊密,也就是說C#裏的不少編程方式均可覺得Autofac使用,例如能夠用Lambda表達式註冊組件
- 較低的學習曲線,學習它很是的簡單,只要你理解了IoC和DI的概念以及在什麼時候須要使用它們
- XML配置支持
- 自動裝配
- 與Asp.Net MVC 3集成
- 微軟的Orchad開源程序使用的就是Autofac,從該源碼能夠看出它的方便和強大
既然它都這麼牛X了,咱們用它就理所固然了,因此推薦其爲IOC的終極解決方案!sql
Autofac是一款IOC框架,比較於其餘的IOC框架,如Spring.NET,Unity,Castle等等所包含的,它很輕量級性能上很是高。編程
官方網站http://autofac.org/oracle
源碼下載地址https://github.com/autofac/Autofac框架
- 控制反轉(IoC/Inverse Of Control): 調用者再也不建立被調用者的實例,由autofac框架實現(容器建立)因此稱爲控制反轉。
- 依賴注入(DI/Dependence injection) : 容器建立好實例後再注入調用者稱爲依賴注入。
中文學習資源:Autofac中文文檔函數
2、AutoFac的使用
首先你必須獲取AutoFac,這裏你能夠經過各類方式加載它,我這裏仍是經過VS中的NuGet來加載AutoFac,不管是哪一種方式,最終的目的就是將 Autofac.dll,Autofac.Configuration.dll 這兩個程序集引用到你的項目中。這樣在你的項目中,若是想使用AutoFac,只需添加其命名空間引用便可~post
一、AutoFac入門
咱們先定義一個數據訪問的接口:性能
public interface IDAL { void Insert(string commandText); }
而後用Sql和Oracle兩種方式分別實現上述接口,不過這裏只是演示而已,因此並無真正去實現這兩個類,你懂的~
SQL方式:
public class SqlDAL : IDAL { public void Insert(string commandText) { Console.WriteLine("使用sqlDAL添加相關信息"); } }
Oracle方式:
public class OracleDAL : IDAL { public void Insert(string commandText) { Console.WriteLine("使用OracleDAL添加相關信息"); } }
而後注入實現構造函數注入:
public class DBManager { IDAL _dal; public DBManager(IDAL dal) { _dal= dal; } public void Add(string commandText) { _dal.Insert(commandText); } }
最後要真正完成依賴注入就得AtuoFac登場了:
var builder = new ContainerBuilder(); builder.RegisterType<DBManager>(); builder.RegisterType<SqlDAL>().As<IDAL>(); using (var container = builder.Build()) { var manager = container.Resolve<DBManager>(); manager.Add("INSERT INTO Persons VALUES ('Man', '25', 'WangW', 'Shanghai')"); }
從以上栗子能夠看出,其實AutoFac的使用跟Unity的使用有點像,關鍵的東東就是這個Container容器類
二、AutoFac經常使用方法說明
(1)builder.RegisterType<Object>().As<Iobject>():註冊類型及其實例。例以下面就是註冊接口IDAL的實例SqlDAL
ContainerBuilder builder = new ContainerBuilder(); builder.RegisterType<SqlDAL>().As<IDAL>(); IContainer container = builder.Build(); SqlDAL sqlDAL = (SqlDAL)container.Resolve<IDAL>();
(2)IContainer.Resolve<IDAL>():解析某個接口的實例。例如上面的最後一行代碼就是解析IDAL的實例SqlDAL
(3)builder.RegisterType<Object>().Named<Iobject>(string name):爲一個接口註冊不一樣的實例。有時候不免會碰到多個類映射同一個接口,好比SqlDAL和OracleDAL都實現了IDAL接口,爲了準確獲取想要的類型,就必須在註冊時起名字。
builder.RegisterType<SqlDAL>().Named<IDAL>("sql"); builder.RegisterType<OracleDAL>().Named<IDAL>("oracle"); IContainer container = builder.Build(); SqlDAL sqlDAL = (SqlDAL)container.ResolveNamed<IDAL>("sql"); OracleDAL oracleDAL = (OracleDAL)container.ResolveNamed<IDAL>("oracle");
(4)IContainer.ResolveNamed<IDAL>(string name):解析某個接口的「命名實例」。例如上面的最後一行代碼就是解析IDAL的命名實例OracleDAL
(5)builder.RegisterType<Object>().Keyed<Iobject>(Enum enum):以枚舉的方式爲一個接口註冊不一樣的實例。有時候咱們會將某一個接口的不一樣實現用枚舉來區分,而不是字符串,例如:
public enum DBType{ Sql, Oracle} builder.RegisterType<SqlDAL>().Keyed<IDAL>(DBType.Sql); builder.RegisterType<OracleDAL>().Keyed<IDAL>(DBType.Oracle); IContainer container = builder.Build(); SqlDAL sqlDAL = (SqlDAL)container.ResolveKeyed<IDAL>(DBType.Sql); 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時給其賦值
DBManager manager = container.Resolve<DBManager>(new NamedParameter("name", "SQL"));
public class DBManager { IDAL dal; public DBManager (string name,IDAL _dal) { Name = name; dal= _dal; } }
(10)builder.RegisterAssemblyTypes(Assembly assembly):註冊程序集下全部類型
builder.RegisterAssemblyTypes(typeof(Program).Assembly).AsImplementedInterfaces();
//或者 AsImplementedInterfaces表示註冊的類型,以接口的方式註冊
builder.RegisterAssemblyTypes(typeof(IRepository<>).Assembly).Where(t => t.IsClass && t.Name.EndsWith("Repository")).AsImplementedInterfaces();
三、經過配置的方式使用AutoFac
(1)先配置好配置文件
<?xml version="1.0"?> <configuration> <configSections> <section name="autofac" type="Autofac.Configuration.SectionHandler, Autofac.Configuration"/> </configSections> <autofac defaultAssembly="ConsoleApplication1"> <components> <component type="ConsoleApplication1.SqlDAL, ConsoleApplication1" service="ConsoleApplication1.IDAL" /> </components> </autofac> </configuration>
(2)讀取配置實現依賴注入(注意引入Autofac.Configuration.dll)
static void Main(string[] args) { ContainerBuilder builder = new ContainerBuilder(); builder.RegisterType<DBManager>(); builder.RegisterModule(new ConfigurationSettingsReader("autofac")); using (IContainer container = builder.Build()) { DBManager manager = container.Resolve<DBManager>(); manager.Add("INSERT INTO Persons VALUES ('Man', '25', 'WangW', 'Shanghai')"); }
3、ASP.NET MVC與AtuoFac
終於到了ASP.NET MVC與AtuoFac雙劍合璧的時候了,下面就看看AtuoFac在MVC中的應用,其實很簡單,大概就幾個步驟搞定:
一、首先在函數Application_Start() 註冊本身的控制器類,必定要引入Autofac.Integration.Mvc.dll
using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Http; using System.Web.Mvc; using System.Web.Optimization; using System.Web.Routing; using Autofac; using AtuoFacOfMVC4.Models; using System.Reflection; using Autofac.Integration.Mvc; namespace AtuoFacOfMVC4 { public class MvcApplication : System.Web.HttpApplication { protected void Application_Start() { var builder = new ContainerBuilder(); SetupResolveRules(builder); builder.RegisterControllers(Assembly.GetExecutingAssembly()); var container = builder.Build(); DependencyResolver.SetResolver(new AutofacDependencyResolver(container)); AreaRegistration.RegisterAllAreas(); WebApiConfig.Register(GlobalConfiguration.Configuration); FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters); RouteConfig.RegisterRoutes(RouteTable.Routes); BundleConfig.RegisterBundles(BundleTable.Bundles); AuthConfig.RegisterAuth(); } private void SetupResolveRules(ContainerBuilder builder) { builder.RegisterType<StudentRepository>().As<IStudentRepository>(); } } }
二、如今在你的MVC程序中注入依賴代碼就ok了
(1)首先聲明一個Student學生類
using System; using System.Collections.Generic; using System.Linq; using System.Web; namespace AtuoFacOfMVC4.Models { public class Student { public int Id { get; set; } public string Name { get; set; } public string Graduation { get; set; } public string School { get; set; } public string Major { get; set; } } }
(2)而後聲明倉儲接口及其實現
using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace AtuoFacOfMVC4.Models { public interface IStudentRepository { IEnumerable<Student> GetAll(); Student Get(int id); Student Add(Student item); bool Update(Student item); bool Delete(int id); } }
(3)最後添加控制器StudentController,並注入依賴代碼
using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; using AtuoFacOfMVC4.Models; namespace AtuoFacOfMVC4.Controllers { public class StudentController : Controller { readonly IStudentRepository repository; //構造器注入 public StudentController(IStudentRepository repository) { this.repository = repository; } public ActionResult Index() { var data = repository.GetAll(); return View(data); } } }
(4)最後爲控制器StudentController的Index方法添加視圖便可,這裏再也不詳述,運行效果以下