[Solution] DI原理解析及Castle、Unity框架使用
本节目录
DI介绍
控制反转(Inversion of Control,英文缩写为IoC)是一个重要的面向对象编程的法则来削减计算机程序的耦合问题.
依赖注入(Dependency Injection,英文缩写为DI)是一种设计模式.
其实本质都是指同一件事,强调的内容不一样.IoC强调容器的作用,DI强调注入的作用.
通常IoC和DI可以理解为一个意思,只是指的对象不同.
DI基本原理
DI本质上是通过容器来反射创建实例.
1个简单的类
class Person
{
public void Say()
{
Console.WriteLine("Person's say method is Called");
}
}
反射代码(className:类的全限定名)
private static object CreateInstance(Assembly assembly, string className)
{
var type = assembly.GetType(className);
return type != null ? Activator.CreateInstance(type) : null;
}
执行(XX为命名空间)
static void Main(string[] args)
{
var obj = CreateInstance(Assembly.GetExecutingAssembly(), "XX.Person");
var person = obj as Person;
if (person != null)
{
person.Say();
}
Console.ReadKey();
}
在上面能看到1个问题,一般情况下.既然使用DI,就不知道具体的注入对象.所以强调面向接口编程.
所以实际上一般先定义接口,再通过DI容器创建对象.
interface IPerson
{
void Say();
}
class Person : IPerson
{
public void Say()
{
Console.WriteLine("Person's say method is Called");
}
}
执行
static void Main(string[] args)
{
var obj = CreateInstance(Assembly.GetExecutingAssembly(), "Demo.Person");
var person = obj as IPerson;
if (person != null)
{
person.Say();
}
Console.ReadKey();
}
DI框架
DI框架流行的有Castle Windsor,Unity...(Autofac Spring.Net已经聊过,不再演示)
在DI框架中,一般需要将对象注册到容器中,然后从容器解析出来.
Castle
Install-Package Castle.Windsor
待注入类
interface ITransient
{ } interface IPerson
{
void Say();
}
class Person : IPerson, ITransient
{
public void Say()
{
Console.WriteLine("Person's say method is Called");
}
}
注册解析方式一
static void Main(string[] args)
{
using (var container = new WindsorContainer())
{
container.Register(Component.For<Person, IPerson>());
var person = container.Resolve<IPerson>();
person.Say();
}
Console.ReadKey();
}
注册解析方式二
public class AssmInstaller : IWindsorInstaller
{
public void Install(IWindsorContainer container, IConfigurationStore store)
{
container.Register(Classes.FromThisAssembly() //选择Assembly
.IncludeNonPublicTypes() //约束Type
.BasedOn<ITransient>() //约束Type
.WithService.DefaultInterfaces() //匹配类型
.LifestyleTransient()); //注册生命周期
}
}
static void Main(string[] args)
{
using (var container = new WindsorContainer())
{
container.Install(new AssmInstaller());
var person = container.Resolve<IPerson>();
person.Say();
}
Console.ReadKey();
}
构造函数注入
class Task : ITransient
{
public IPerson Person { get; set; }
public Task(IPerson person)
{
Person = person;
Person.Say();
}
}
static void Main(string[] args)
{
using (var container = new WindsorContainer())
{
container.Install(new AssmInstaller());
container.Resolve<Task>();
}
Console.ReadKey();
}
属性注入
class Task : ITransient
{
public IPerson Person { get; set; }
public Task()
{
}
public void Say()
{
Person.Say();
}
}
static void Main(string[] args)
{
using (var container = new WindsorContainer())
{
container.Install(new AssmInstaller());
container.Resolve<Task>().Say();
}
Console.ReadKey();
}
MVC集成
Install-Package Castle.Windsor.Mvc
Application_Start注册
protected void Application_Start()
{
RouteConfig.RegisterRoutes(RouteTable.Routes);
var container = new WindsorContainer()
.Install(FromAssembly.This());
var controllerFactory = new WindsorControllerFactory(container.Kernel);
ControllerBuilder.Current.SetControllerFactory(controllerFactory);
}
Installer注册
public class AssmInstaller : IWindsorInstaller
{
public void Install(IWindsorContainer container, IConfigurationStore store)
{
container.Register(Classes.FromThisAssembly()
.IncludeNonPublicTypes()
.BasedOn<ITransient>()
.WithService.DefaultInterfaces()
.LifestyleTransient());
container.Register(Classes.FromThisAssembly()
.BasedOn<Controller>()
.LifestyleTransient()
);
}
}
这样Castle Windsor就能接管解析Controller了.
Unity
Install-Package Unity
待注入类
public interface IPerson
{
void Say();
}
public class Person : IPerson
{
public void Say()
{
Console.WriteLine("Person's say method is Called");
}
}
注册解析一
static void Main(string[] args)
{
using (var container = new UnityContainer())
{
container.RegisterType<IPerson, Person>(new TransientLifetimeManager());
var person = container.Resolve<IPerson>();
person.Say();
}
Console.ReadKey();
}
注册解析二
static void Main(string[] args)
{
using (var container = new UnityContainer())
{
container.RegisterInstance<IPerson>(new Person());
var person = container.Resolve<IPerson>();
person.Say();
}
Console.ReadKey();
}
构造函数注入
class Task : ITask
{
public IPerson Person { get; set; }
public Task(IPerson person)
{
Person = person;
Person.Say();
}
} public interface ITask
{ }
static void Main(string[] args)
{
using (var container = new UnityContainer())
{
container.RegisterInstance<IPerson>(new Person());
container.RegisterType<ITask, Task>();
container.Resolve<ITask>();
}
Console.ReadKey();
}
属性注入
class Task : ITask
{
[Dependency]
public IPerson Person { get; set; }
public Task(IPerson person)
{
Person = person;
}
public void Say()
{
Person.Say();
}
}
static void Main(string[] args)
{
using (var container = new UnityContainer())
{
container.RegisterInstance<IPerson>(new Person());
container.RegisterType<ITask, Task>();
var task = container.Resolve<ITask>();
task.Say();
}
Console.ReadKey();
}
MVC集成
Install-Package Unity.Mvc
Application_Start注册
protected void Application_Start()
{
RouteConfig.RegisterRoutes(RouteTable.Routes);
var container = new UnityContainer();
container.RegisterType<IPerson, Person>();
DependencyResolver.SetResolver(new UnityDependencyResolver(container));
}
这样Unity就接管了Controller的创建
[Solution] DI原理解析及Castle、Unity框架使用的更多相关文章
- [Solution] AOP原理解析及Castle、Autofac、Unity框架使用
本节目录: AOP介绍 AOP基本原理 AOP框架 Castle Core Castle Windsor Autofac Unity AOP介绍 面向切面编程(Aspect Oriented Prog ...
- AOP原理解析及Castle、Autofac、Unity框架使用
转自:https://www.cnblogs.com/neverc/p/5241466.html AOP介绍 面向切面编程(Aspect Oriented Programming,英文缩写为AOP), ...
- DI 原理解析 并实现一个简易版 DI 容器
本文基于自身理解进行输出,目的在于交流学习,如有不对,还望各位看官指出. DI DI-Dependency Injection,即"依赖注入":对象之间依赖关系由容器在运行期决定, ...
- [置顶]
滴滴插件化框架VirtualAPK原理解析(一)之插件Activity管理
上周末,滴滴与360都开源了各自的插件化框架,VirtualAPK与RePlugin,作为一个插件化方面的狂热研究者,在周末就迫不及待的下载了Virtualapk框架来进行研究,本篇博客带来的是Vir ...
- ABP中动态WebAPI原理解析
ABP中动态WebAPI原理解析 动态WebAPI应该算是ABP中最Magic的功能之一了吧.开发人员无须定义继承自ApiController的类,只须重用Application Service中的类 ...
- Spring IOC设计原理解析:本文乃学习整理参考而来
Spring IOC设计原理解析:本文乃学习整理参考而来 一. 什么是Ioc/DI? 二. Spring IOC体系结构 (1) BeanFactory (2) BeanDefinition 三. I ...
- Web APi之过滤器创建过程原理解析【一】(十)
前言 Web API的简单流程就是从请求到执行到Action并最终作出响应,但是在这个过程有一把[筛子],那就是过滤器Filter,在从请求到Action这整个流程中使用Filter来进行相应的处理从 ...
- JavaScript 模板引擎实现原理解析
1.入门实例 首先我们来看一个简单模板: <script type="template" id="template"> <h2> < ...
- Request 接收参数乱码原理解析二:浏览器端编码原理
上一篇<Request 接收参数乱码原理解析一:服务器端解码原理>,分析了服务器端解码的过程,那么浏览器是根据什么编码的呢? 1. 浏览器解码 浏览器根据服务器页面响应Header中的“C ...
随机推荐
- Linux驱动开发学习笔记(1):LINUX驱动版本的hello world
1.关于目录 /lib/modules/2.6.9-42.ELsmp/build/ 这个是内核源码所在的目录 一般使用这样的命令进入这个目录:cd /lib/modules/$(una ...
- Apache Storm 的历史及经验教训——Nathan Marz【翻译】
英文原文地址 中英文对照地址 History of Apache Storm and lessons learned --项目创建者 Nathan Marz Apache Storm 最近成为了ASF ...
- Spark使用总结与分享
背景 使用spark开发已有几个月.相比于python/hive,scala/spark学习门槛较高.尤其记得刚开时,举步维艰,进展十分缓慢.不过谢天谢地,这段苦涩(bi)的日子过去了.忆苦思甜,为了 ...
- SSH乱码解决
解决方案: 使用linux,在用户根目录(/root)下有一个.bash_profile配置文件,该配置只对当前用户有效. 使用ls -a命令可以查看到该文件.使用vi编辑器打开该文件后,在其中加入 ...
- Cocos2dx使用wxsqlite开源加密SQLite3数据库
最近使用wxsqlite加密sqlite3数据库,刚开始折腾好几天,在xcode上一直编译不通过,后来在sqlite3.c找到配置,编译顺利通过,太激动了,哈哈,废话少说!总结一下android和io ...
- Tomcat抛出异常:ClientAbortException: java.net.SocketException: Connection
在做一个小网站的时候,写了一个通过servlet实现文件下载功能的页面.当我点击超级练级,弹出下载对话框,点击保存正常下载,不会出现任何问题,当我我点击取消,服务器端就出现如下提示: ClientAb ...
- 自定义android RadioButton View,添加较为灵活的布局处理方式
android的RadioButton的使用历来都让人比较头疼,如在布局方面,图案.文字无法分别设置padding等,另外,低版本的android RadioGroup不支持换行排列的RadioBut ...
- Flink 案例整合
1.概述 Flink 1.1.0 版本已经在官方发布了,官方博客于 2016-08-08 更新了 Flink 1.1.0 的变动.在这 Flink 版本的发布,添加了 SQL 语法这一特性.这对于业务 ...
- u3d动态加入模型
楼层一层一层的加,把模型分开,弄成prefab放到Resourse文件夹里,在代码里用Instantiate(Resources.Load("模型名字") as GameObjec ...
- aspose.cell 设置excel里面的文字是超链接
目的: 1.通过方法designer.Workbook.Worksheets[0].Hyperlinks.Add("A1", 1, 1, url);给导出到excel里面的数据加上 ...