ASP.NET IOC之 AutoFac的认识和结合MVC的使用
这几天研究了解发现AutoFac是个牛X的IOC容器,是.NET领域比较流行的IOC框架之一,传说是速度最快的,~ 据相关资料,相关学习,和认知,遂做了一些整理
优点:
它是C#语言联系很紧密,也就是说C#里的很多编程方式都可以为Autofac使用,例如可以用Lambda表达式注册组件
较低的学习曲线,学习它非常的简单,只要你理解了IoC和DI的概念以及在何时需要使用它们
XML配置支持
自动装配
与Asp.Net MVC 集成
微软的Orchad开源程序使用的就是Autofac,从该源码可以看出它的方便和强大
还有相关介绍:http://www.oschina.net/p/autofac
有博友做了相关测试:http://www.cnblogs.com/liping13599168/archive/2011/07/17/2108734.html
所以墙裂推荐用他作为IOC的终极解决方案!
一、AutoFac的简单使用,第二点介绍在MVC中的使用
将Autofac整合到你的应用的基本模式如下:
- 按照控制反转(IoC)的思想构建你的应用程序
- 添加Autofac引用
- 在application启动代码里创建
ContainerBuilder
对象 - 注册组件
- Build容器并且保存以备用
- 在程序执行阶段
- 从容器创建一个作用域
- 在作用域里获取组件的实例
首先获取AutoFac,这里你可以通过各种方式下载,刚开始我找的各种地址都失效了,有的也没更新,各种不靠谱,
推荐直接用VS中的NuGet来加载AutoFac,(路径:工具——NuGet包管理器——管理解决方案的NuGet的程序包——搜索框输入AutoFac即可,ps搜索框上面得是浏览)
不论是哪种方式,最终的目的就是将 Autofac.dll,Autofac.Configuration.dll 这两个程序集引用到你的项目中。
这样在你的项目中,如果想使用AutoFac,只需添加其命名空间引用即可~
控制反转的思想是,在类的构造函数里传递依赖而不是在应用程序里将这些类绑在一起让类自己创建他们的依赖。关于依赖注入
http://www.cnblogs.com/zhangchenliang/archive/2013/01/08/2850970.html这里有一个很形象的例子。如果感兴趣可以去了解一下。
1、新建项目,引入DLL或者通过NuGet安装,然后定义一个数据访问的接口:
// 这个接口帮助我们从Console 解耦"输出"方法
// 我们不需要关心怎样输出,只要知道能输出即可
public interface IDAL
{
void Insert(string commandText);
}
2、用Sql和Oracle两种方式分别实现以上接口(仅演示,不是真的实现数据交互)
// 这里IDAL接口的实现完成向控制台的输出。
public class SqlDAL : IDAL
{
public void Insert(string commandText)
{
Console.WriteLine("使用sql添加相关信息");
}
} public class OracleDAL : IDAL
{
public void Insert(string commandText)
{
Console.WriteLine("使用Oracle添加相关信息");
}
}
3、实现构造函数依赖
// DBManager 对上述接口依赖
// 注意这里的构造函数参数是IDAL 类型
// 这样这个Add 的Insert方法由IDAL 的实现决定 public class DBManager
{
IDAL _dal;
public DBManager(IDAL dal)
{
_dal= dal;
} public void Add(string commandText)
{
_dal.Insert(commandText);
}
}
4、AtuoFac的使用
说明: 在应用程序启动过程中,首先要创建一个ContainerBuilder
并且用它注册你的组件。组件是一个表达式, .NET类型或者其他的一段暴露一个或者多个服务的代码并且可以用在其他的依赖里。
简单来说,就像下面的例子所述,定义一个实现某个接口的.NET类型
public class SomeType : IService
{
}
你可以通过下面两种方法之一来定位这个类型
- 指定类型本身,SomeType
- 指定接口,一个IService
在这个例子里,这个组件是SomeType
,他暴露的服务是SomeType
和IService
.
在Autofac里,你应当用一个ContainerBuilder
来注册他们,如下:
static void Main(string[] args)
{
// 创建你的builder
var builder = new ContainerBuilder();
//注册DBManager,当注册的类型在相应得到的容器中可以Resolve(解析)DBManager实例。
builder.RegisterType<DBManager>();
// 注册类型及其实例。例如下面就是注册接口IDAL的实例OracleDAL,通过AS可以让DBManager类中通过构造函数依赖注入类型相应的接口。
builder.RegisterType<OracleDAL>().As<IDAL>();
builder.RegisterType<SqlDAL>().As<IDAL>(); // 创建作用域,Build()方法生成一个对应的Container实例使用,然后释放
using (var container = builder.Build())
{
//通过Resolve解析到注册的类型实例。
var manager = container.Resolve<DBManager>();
manager.Add("INSERT INTO Persons VALUES ('Man', '25', 'WangW', 'Shanghai')");
}
Console.ReadKey();
}
当你运行你的程序时…
- Add()方法要求Autofac创建了一个DBManager对象
- Autofac发现DBManager构造函数需要IDAL
- Autofac发现IDAL 映射到SqlDAL,所以创建一个新的SqlDAL的实例
- Autofac使用SqlDAL实例完成IDAL 的创建
- Autofac返回一个完整的DBManager对象给”Add()”来消费
输出结果:“使用sql添加相关信息”,
如果希望输出不同的内容,改变一下注册过程即可实现控制反转了
输出:"使用Oracle添加相关信息"
这是比较简单的使用方式,如果复杂点,可以看看这篇
http://www.cnblogs.com/liping13599168/archive/2011/07/16/2108209.html
二、AtuoFac AtuoFac在ASP.NET MVC中的应用
1、首先在函数Application_Start() 注册自己的控制器类,一定要引入Autofac.Integration.Mvc.dll
using System;
using System.Reflection;
using System.Web;
using System.Web.Mvc;
using System.Web.Routing;
using System.Web.Optimization;
using Autofac;
using Autofac.Integration.Mvc;
using AutofacDemo.Models; namespace AutofacDemo
{
public class MvcApplication : System.Web.HttpApplication
{
protected void Application_Start()
{
//依赖注入开始
var builder = new ContainerBuilder();
builder.RegisterType<StudentRepository>().As<IStudentRepository>();//注册类型及其实例
builder.RegisterControllers(Assembly.GetExecutingAssembly());//注册当前程序集内所有的Controller类。
//// builder.RegisterAssemblyTypes(Assembly.GetExecutingAssembly()).AsImplementedInterfaces();//注册当前程序集内的所有类
var container = builder.Build();
DependencyResolver.SetResolver(new AutofacDependencyResolver(container)); //MVC用到的
AreaRegistration.RegisterAllAreas();
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
RouteConfig.RegisterRoutes(RouteTable.Routes);
BundleConfig.RegisterBundles(BundleTable.Bundles);
}
}
}
2、首先声明一个Student学生类,接口以及实现
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
namespace AutofacDemo.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; }
} ///声明仓储接口仓储接口及其实现
public interface IStudentRepository
{
IEnumerable<Student> GetAll();
Student Get(int id);
Student Add(Student item);
bool Update(Student item);
bool Delete(int id);
}
public class StudentRepository : IStudentRepository
{
private List<Student> Articles = new List<Student>(); public StudentRepository()
{
//添加演示数据
Add(new Student { Id = , Name = "张三", Major = "软件工程", Graduation = "2013年", School = "西安工业大学" });
Add(new Student { Id = , Name = "李四", Major = "计算机科学与技术", Graduation = "2013年", School = "西安工业大学" });
Add(new Student { Id = , 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 == -)
{
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;
}
}
}
3、添加控制器StudentController,并注入依赖代码
public class StudentController : Controller
{
readonly IStudentRepository _repository; //构造器注入
public StudentController(IStudentRepository repository)
{
this._repository = repository;
} // GET: Student
public ActionResult Index()
{
var data = _repository.GetAll();
return View(data);
}
}
4、为控制器StudentController的Index方法添加视图
@{
ViewBag.Title = "Index";
} <h2>Index</h2> @model List<AutofacDemo.Models.Student>
<table id="datatable" cellpadding="" cellspacing="">
<thead class="gray_d">
<tr>
<td class="sth" style="width:3%;">序号</td>
<td class="sth" style="width:4%;">姓名</td>
<td class="sth" style="width:4%;">专业</td>
<td class="sth" style="width:4%;">日期</td>
<td class="sth" style="width:4%;">学校</td>
</tr>
</thead>
<tbody>
@foreach (var item in Model)
{
<tr target="sid_user" rel="@Html.DisplayFor(modelItem => item.Id)">
<td> @Html.DisplayFor(modelItem => item.Id) </td>
<td> @Html.DisplayFor(modelItem => item.Name) </td>
<td> @Html.DisplayFor(modelItem => item.Major) </td>
<td> @Html.DisplayFor(modelItem => item.Graduation) </td>
<td> @Html.DisplayFor(modelItem => item.School) </td>
<td>
</tr>
}
</tbody>
</table>
三、关于AutoFac的一些常用方法说明
、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>(); 、IContainer.Resolve<IDAL>()://解析某个接口的实例。例如上面的最后一行代码就是解析IDAL的实例SqlDAL 、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"); 、IContainer.ResolveNamed<IDAL>(string name)://解析某个接口的“命名实例”。例如上面的最后一行代码就是解析IDAL的命名实例OracleDAL 、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);
、IContainer.ResolveKeyed<IDAL>(Enum enum): //根据枚举值解析某个接口的特定实例。例如上面的最后一行代码就是解析IDAL的特定实例OracleDAL 、builder.RegisterType<Worker>().InstancePerDependency()://用于控制对象的生命周期,每次加载实例时都是新建一个实例,默认就是这种方式 、builder.RegisterType<Worker>().SingleInstance(): //用于控制对象的生命周期,每次加载实例时都是返回同一个实例 、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 (使用方式差不多,需要配置,比上面麻烦点,主要优势 例如实现数据库动态切换,避免程序重新编译,然并卵,谁没事换个数据库玩 先贴上)
()修改App.config配置文件 一定要注意新建项目 会默认有这个
<startup>
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.5.2" />
</startup>
//不能让他出现在<configuration> 下第一个节点,不然就报错,
<?xml version="1.0"?>
<configuration>
<configSections>
<section name="autofac" type="Autofac.Configuration.SectionHandler, Autofac.Configuration"/>
</configSections>
<autofac defaultAssembly="AutofacDemo">
<components>
<component type="AutofacDemo.SqlDatabase, AutofacDemo" service="AutofacDemo.IDatabase" />
</components>
</autofac>
</configuration> 通过Autofac.Configuration.SectionHandler配置节点对组件进行处理。 ()读取配置实现依赖注入(注意引入Autofac.Configuration.dll) static void Main(string[] args)
{
var builder = new ContainerBuilder();
builder.RegisterType<DBManager>();
builder.RegisterModule(new ConfigurationSettingsReader("autofac"));
using (var container = builder.Build())
{
var manager = container.Resolve<DBManager>();
manager.Add("INSERT INTO Persons VALUES ('Man', '25', 'WangW', 'Shanghai')");
}
}
源码共享地址 : \\10.100.27.47\yebo750\Demo (打开我的电脑输入即可)
ASP.NET IOC之 AutoFac的认识和结合MVC的使用的更多相关文章
- IoC容器Autofac(5) - Autofac在Asp.net MVC Filter中的应用
Autofac结合EF在MVC中的使用,上一篇IoC容器Autofac(4) - Autofact + Asp.net MVC + EF Code First(附源码)已经介绍了.但是只是MVC中Co ...
- [转]ASP.NET MVC IOC 之AutoFac攻略
本文转自:http://www.cnblogs.com/WeiGe/p/3871451.html 一.为什么使用AutoFac? 之前介绍了Unity和Ninject两个IOC容器,但是发现园子里用A ...
- ASP.NET MVC IOC 之AutoFac攻略
一.为什么使用AutoFac? 之前介绍了Unity和Ninject两个IOC容器,但是发现园子里用AutoFac的貌似更为普遍,于是捯饬了两天,发现这个东东确实是个高大上的IOC容器~ Autofa ...
- IoC容器Autofac - Autofac + Asp.net MVC + EF Code First(转载)
转载地址:http://www.cnblogs.com/JustRun1983/archive/2013/03/28/2981645.html 有修改 Autofac通过Controller默认构造 ...
- ASP.NET MVC IOC 之AutoFac
ASP.NET MVC IOC 之AutoFac攻略 一.为什么使用AutoFac? 之前介绍了Unity和Ninject两个IOC容器,但是发现园子里用AutoFac的貌似更为普遍,于是捯饬了两天, ...
- Ioc容器Autofac系列(2)-- asp.net mvc中整合autofac
经过上篇蜻蜓点水的介绍后,本篇通过实例快速上手autofac,展示当asp.net mvc引入了autofac之后会带来什么. 创建Asp.net MVC并引入Autofac 首先,创建一个MVC站点 ...
- [转]Ioc容器Autofac
本文转自:http://www.cnblogs.com/hkncd/archive/2012/11/21/2780041.html Ioc容器Autofac系列(1)-- 初窥 前言 第一次接触A ...
- Ioc容器Autofac系列(1)-- 初窥
一.前言 第一次接触Autofac是因为CMS系统--Orchard,后来在一个开源爬虫系统--NCrawler中也碰到过,随着深入了解,我越发觉得Ioc容器是Web开发中必不可少的利器.那么,Io ...
- .net core2.0下Ioc容器Autofac使用
.net core发布有一段时间了,最近两个月开始使用.net core2.0开发项目,大大小小遇到了一些问题.准备写个系列介绍一下是如何解决这些问题以及对应技术.先从IOC容器Autofac开始该系 ...
随机推荐
- Leetcode题库——9.回文数
@author: ZZQ @software: PyCharm @file: HuiWenShu.py @time: 2018/9/16 16:51 要求:判断一个整数是否是回文数.回文数是指正序(从 ...
- ACM的fflush(stdin)的问题
在最近的刷题过程中,因为用到了很多字符串的操作,有时需要多次清空缓冲区,所以用了fflush(stdin);的语句,确实很好用,但是发现在OJ上提交后会出现runtime error的问题.当时并没有 ...
- 微信小程序公共组件的引用与控制
思路: 1.在组件wxml文件里实现布局.数据绑定.事件绑定: 2.组件js文件里定义事件,并将文件所有内容作为一个对象export出去:3.在引用的文件引入组件(方式有两种,一个是用include引 ...
- PAT 1060 爱丁顿数
https://pintia.cn/problem-sets/994805260223102976/problems/994805269312159744 英国天文学家爱丁顿很喜欢骑车.据说他为了炫耀 ...
- pygame学习笔记(3)——时间、事件、文字
转载请注明:@小五义 http://www.cnblogs.com/xiaowuyi 1.运动速率 上节中,实现了一辆汽车在马路上由下到上行驶,并使用了pygame.time.delay(200 ...
- Linux内核0.11 bootsect文件说明
一.总体功能介绍 这是关于Linux-kernel-0.11中boot文件夹下bootsect.s源文件的说明,其中涉及到了一些基础知识可以参考这两篇文章. 操作系统启动过程 软盘相关知识和通过BIO ...
- Android 目录结构
Android目录结构中,values目录下对应的是应用程序所需要的数据,网上看到了一个包含values-v11等values-*的写法. 是为了进行分辨率的自适应????????? 因为还没有涉及到 ...
- SpringBoot(十三)_springboot上传Excel并读取excel中的数据
今天工作中,发现同事在整理数据,通过excel上传到数据库.所以现在写了篇利用springboot读取excel中的数据的demo.至于数据的进一步处理,大家肯定有不同的应用场景,自行修改 pom文件 ...
- spring 整合 Struts1.X [转]
这篇博客摘自[http://blog.csdn.net/chendc201/article/details/8464008], 其中也有一些是自己增加的部分 . 第一步, 需要为 Struts 装载 ...
- 第219天:Angular---过滤器
在Angular中,过滤器的功能主要是格式化数据表达式,且可以自定义过滤器.作用域(scope)主要服务于页面模板,在控制器和页面中起桥梁作用,保存模板中的数据对象,为模板中的元素提供方法和属性. 一 ...