本节目录

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框架使用的更多相关文章

  1. [Solution] AOP原理解析及Castle、Autofac、Unity框架使用

    本节目录: AOP介绍 AOP基本原理 AOP框架 Castle Core Castle Windsor Autofac Unity AOP介绍 面向切面编程(Aspect Oriented Prog ...

  2. AOP原理解析及Castle、Autofac、Unity框架使用

    转自:https://www.cnblogs.com/neverc/p/5241466.html AOP介绍 面向切面编程(Aspect Oriented Programming,英文缩写为AOP), ...

  3. DI 原理解析 并实现一个简易版 DI 容器

    本文基于自身理解进行输出,目的在于交流学习,如有不对,还望各位看官指出. DI DI-Dependency Injection,即"依赖注入":对象之间依赖关系由容器在运行期决定, ...

  4. [置顶] 滴滴插件化框架VirtualAPK原理解析(一)之插件Activity管理

    上周末,滴滴与360都开源了各自的插件化框架,VirtualAPK与RePlugin,作为一个插件化方面的狂热研究者,在周末就迫不及待的下载了Virtualapk框架来进行研究,本篇博客带来的是Vir ...

  5. ABP中动态WebAPI原理解析

    ABP中动态WebAPI原理解析 动态WebAPI应该算是ABP中最Magic的功能之一了吧.开发人员无须定义继承自ApiController的类,只须重用Application Service中的类 ...

  6. Spring IOC设计原理解析:本文乃学习整理参考而来

    Spring IOC设计原理解析:本文乃学习整理参考而来 一. 什么是Ioc/DI? 二. Spring IOC体系结构 (1) BeanFactory (2) BeanDefinition 三. I ...

  7. Web APi之过滤器创建过程原理解析【一】(十)

    前言 Web API的简单流程就是从请求到执行到Action并最终作出响应,但是在这个过程有一把[筛子],那就是过滤器Filter,在从请求到Action这整个流程中使用Filter来进行相应的处理从 ...

  8. JavaScript 模板引擎实现原理解析

    1.入门实例 首先我们来看一个简单模板: <script type="template" id="template"> <h2> < ...

  9. Request 接收参数乱码原理解析二:浏览器端编码原理

    上一篇<Request 接收参数乱码原理解析一:服务器端解码原理>,分析了服务器端解码的过程,那么浏览器是根据什么编码的呢? 1. 浏览器解码 浏览器根据服务器页面响应Header中的“C ...

随机推荐

  1. ZooKeeper快速搭建

    原文地址:http://nileader.blog.51cto.com/1381108/795230 下载PDF版本 本文是ZooKeeper的快速搭建,旨在帮助大家以最快的速度完成一个ZK集群的搭建 ...

  2. Why数学图像生成工具

    该软件能够以给定的数学公式及算法生成各种绚烂的数学图像.软件中有两种生成图像的方法: (1)通过一种我自定义的脚本语言生成: 软件中定义一套简单易学的脚本语言,用于描述数学表达式.使用时需要先要将数学 ...

  3. ABAP报表中负值展示问题的处理方法

    现象描述 在使用ABAP报表展示数据的时候会涉及到金额类字段,在手动计算金额的时候,有时会发生存在负值而无法正常展示的情况.  处理过程 ABAP报表的数据展示常用的方法有两种,分别是表控制和ALV ...

  4. [转] Linux学习之CentOS(三十六)--FTP服务原理及vsfptd的安装、配置

    本篇随笔将讲解FTP服务的原理以及vsfptd这个最常用的FTP服务程序的安装与配置... 一.FTP服务原理 FTP(File Transfer Protocol)是一个非常古老并且应用十分广泛的文 ...

  5. Visual Studio 2010配置OpenGL-1.8

    参考博客 : 安装参考 1. http://blog.csdn.net/mooncircle/article/details/5545448 2. http://www.cnblogs.com/moo ...

  6. MSSql使用SQL语句快速查看表对的就说明,及表字段描述及字段类型

    --表描述 SELECT tbs.name 表名,ds.value 描述 FROM sys.extended_properties ds LEFT JOIN sysobjects tbs ON ds. ...

  7. pecl install imagick

    steven@server:/var/www$ sudo pecl install imagickdownloading imagick-2.3.0.tgz ...Starting to downlo ...

  8. ios辅助功能之voiceover实战

      一个元素朗读的内容可分为以下4个部分(4部分按先后顺序朗读) 1. Label:元素的标题 2. Value:元素的值(可选) 3. Traits:元素的特征,即类型,包含: 按钮/链接/搜索框/ ...

  9. Oracle数据库入门——pctfree和pctused详解

    一.建立表时候,注意PCTFREE参数的作用 PCTFREE:为一个块保留的空间百分比,表示数据块在什么情况下可以被insert,默认是10,表示当数据块的可用空间低于10%后,就不可以被insert ...

  10. centos6.5+jexus5.6.3+mono 3.10实践,让asp.net在linux上飞一会儿

    备忘,这是给自己看的,用ubuntu server装mono 3.10老是卡在了编译libgdiplus上面,从来就没成功过,郁闷啊,零零散散搞了好几天,作罢.后来试了OpenSUSE 11很容易搞好 ...