ASP.NET MVC IOC 之AutoFac攻略

1、爲何使用AutoFac?

以前介紹了Unity和Ninject兩個IOC容器,可是發現園子裏用AutoFac的貌似更爲廣泛,因而捯飭了兩天,發現這個東東確實是個高大上的IOC容器~sql

Autofac是.NET領域最爲流行的IOC框架之一,傳說是速度最快的一個:編程

優勢: oracle

  • 它是C#語言聯繫很緊密,也就是說C#裏的不少編程方式均可覺得Autofac使用,例如能夠用Lambda表達式註冊組件
  • 較低的學習曲線,學習它很是的簡單,只要你理解了IoC和DI的概念以及在什麼時候須要使用它們
  • XML配置支持
  • 自動裝配
  • 與Asp.Net MVC 3集成
  • 微軟的Orchad開源程序使用的就是Autofac,從該源碼能夠看出它的方便和強大

既然它都這麼牛X了,咱們用它就理所固然了,因此推薦其爲IOC的終極解決方案!框架

2、AutoFac的使用

首先你必須獲取AutoFac,這裏你能夠經過各類方式加載它,我這裏仍是經過VS中的NuGet來加載AutoFac,不管是哪一種方式,最終的目的就是將 Autofac.dll,Autofac.Configuration.dll 這兩個程序集引用到你的項目中。這樣在你的項目中,若是想使用AutoFac,只需添加其命名空間引用便可~ide

一、AutoFac入門

咱們先定義一個數據訪問的接口:函數

public interface IDAL
{
    void Insert(string commandText);
}

而後用Sql和Oracle兩種方式分別實現上述接口,不過這裏只是演示而已,因此並無真正去實現這兩個類,你懂的~學習

SQL方式:ui

public class SqlDAL : IDAL
{
    public void Insert(string commandText)
    {
        Console.WriteLine("使用sqlDAL添加相關信息");
    }
}

Oracle方式:this

public class OracleDAL : IDAL
{
    public void Insert(string commandText)
    {
        Console.WriteLine("使用OracleDAL添加相關信息");
    }
}

而後注入實現構造函數注入:spa

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;
    }
}

三、經過配置的方式使用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);
    }
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;

namespace AtuoFacOfMVC4.Models
{
    public class StudentRepository : IStudentRepository
    {
        private List<Student> Articles = new List<Student>();

        public StudentRepository()
        {
            //添加演示數據
            Add(new Student { Id = 1, Name = "張三", Major = "軟件工程", Graduation = "2013年", School = "西安工業大學" });
            Add(new Student { Id = 2, Name = "李四", Major = "計算機科學與技術", Graduation = "2013年", School = "西安工業大學" });
            Add(new Student { Id = 3, Name = "王五", Major = "自動化", Graduation = "2013年", School = "西安工業大學" });
        }
        /// <summary>
        /// 獲取所有學生信息
        /// </summary>
        /// <returns></returns>
        public IEnumerable<Student> GetAll()
        {
            return Articles;
        }
        /// <summary>
        /// 經過ID獲取學生信息
        /// </summary>
        /// <param name="id"></param>
        /// <returns></returns>
        public Student Get(int id)
        {
            return Articles.Find(p => p.Id == id);
        }
        /// <summary>
        /// 添加學生信息
        /// </summary>
        /// <param name="item"></param>
        /// <returns></returns>
        public Student Add(Student item)
        {
            if (item == null)
            {
                throw new ArgumentNullException("item");
            }
            Articles.Add(item);
            return item;
        }
        /// <summary>
        /// 更新學生信息
        /// </summary>
        /// <param name="item"></param>
        /// <returns></returns>
        public bool Update(Student item)
        {
            if (item == null)
            {
                throw new ArgumentNullException("item");
            }

            int index = Articles.FindIndex(p => p.Id == item.Id);
            if (index == -1)
            {
                return false;
            }
            Articles.RemoveAt(index);
            Articles.Add(item);
            return true;
        }
        /// <summary>
        /// 刪除學生信息
        /// </summary>
        /// <param name="id"></param>
        /// <returns></returns>
        public bool Delete(int id)
        {
            Articles.RemoveAll(p => p.Id == id);
            return true;
        }
    }
}
View Code

(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方法添加視圖便可,這裏再也不詳述,運行效果以下

相關文章
相關標籤/搜索