使用Microsoft.Practices.Unity 依赖注入
Unity是微软Patterns & Practices团队所开发的一个轻量级的,并且可扩展的依赖注入(Dependency Injection)容器,它支持常用的三种依赖注入方式:构造器注入(Constructor Injection)、属性注入(Property Injection),以及方法调用注入(Method Call Injection).
假设我们有下面的场景代码,在代码里面有一个很简单的customer对象,customer 对象有个save 方法, 这个方法通过调用ICustomerDataAccess.Save将数据持久到数据库中, 在列子中我们实现了dataaccess的sql版本和mysql版本,也就是说我们这个customer对象,可以支持持久化到sql server 或 mysql.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Microsoft.Practices.Unity; namespace UnitySample
{
public interface ICustomerDataAccess
{
void Save(Customer c);
} public class CustomerSqlDataAccess : ICustomerDataAccess
{
public void Save(Customer c)
{
Console.Write("{2}, save data id:{0},name{1}",c.Id,c.Name,this.GetType().ToString());
}
} public class CustomerMysqlDataAccess : ICustomerDataAccess
{
public void Save(Customer c)
{
Console.Write("{2}, save data id:{0},name{1}", c.Id, c.Name, this.GetType().ToString());
}
} public class Customer
{
public ICustomerDataAccess CustomerDataAccess { get; set; } public string Id { get; set; }
public string Name { get; set; } public void Save()
{
CustomerDataAccess.Save(this);
}
} class Program
{
static void Main(string[] args)
{ }
}
}
传统做法可能是在配置文件中填几个一个变量 dbType = sql or my sql. 然后在customer 对象中根据设定的db type 来实力化不同的dataaccess.
代码可能会是下面这个样子,这样的话,customer 对象实际上依赖于CustomerSqlDataAccess 和 CustomerSqlDataAccess 的,而且,未来比如要新增加其他数据库的支持,则必须修改customer 对象的源代码。
public Customer()
{
if (DbType = "sql")
{
CustomerDataAccess = new CustomerSqlDataAccess();
}
else
{
CustomerDataAccess = new CustomerSqlDataAccess();
}
}
下面我们使用unity 来解除customer 对象对 CustomerSqlDataAccess 和 CustomerSqlDataAccess 的依赖。
一、 属性注入(Property Injection)
1. 配置unity
为了以后调用方便,我这里建立了一个静态方法,然后注入了两个dataaccess。
public class UnitySetup
{
public static void Config()
{
var container = new UnityContainer();
container.RegisterType<ICustomerDataAccess, CustomerSqlDataAccess>();
container.RegisterType<ICustomerDataAccess, CustomerSqlDataAccess>("mysql");
}
}
然后我们需要在Main函数中调用这个方法配置unity containter.
static void Main(string[] args)
{
UnitySetup.Config();
}
2. 标记属性注入
这步很简单,直接在customer 类中的ICustomerDataAccess 定义上面添加[Dependency]标注即可。
[Dependency]
public ICustomerDataAccess CustomerDataAccess { get; set; }
3. 通过resove 获取对象
var container = new UnityContainer();
UnitySetup.Config(container);
var sqlCustomer = container.Resolve<ICustomerDataAccess>();
var mysqlCustomer = container.Resolve<ICustomerDataAccess>("mysql");
上面的代码就分别获取了CustomerSqlDataAccess 实例和CustomerSqlDataAccess实例
二 、构造器注入(Constructor Injection)
1. 配置unity
unityContainer.RegisterType<Customer>(
new InjectionConstructor(new ResolvedParameter<ICustomerDataAccess>()));
unityContainer.RegisterType<Customer>("mysqlCustomer",
new InjectionConstructor(new ResolvedParameter<ICustomerDataAccess>("mysql")));
添加两行注册customer 对想到container.
2. 获取
var sqlCustomer = container.Resolve<Customer>();
var myqlCustomer = container.Resolve<Customer>("mysqlCustomer"); 完整代码如下
using System;
using System.Collections.Generic;
using System.Data;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Microsoft.Practices.Unity; namespace UnitySample
{ public interface ICustomerDataAccess
{
void Save(Customer c);
} public class CustomerSqlDataAccess : ICustomerDataAccess
{
public void Save(Customer c)
{
Console.WriteLine("{2}, save data id:{0},name{1}",c.Id,c.Name,this.GetType().ToString());
}
} public class CustomerMysqlDataAccess : ICustomerDataAccess
{
public void Save(Customer c)
{
Console.WriteLine("{2}, save data id:{0},name{1}", c.Id, c.Name, this.GetType().ToString());
}
} public class UnitySetup
{
public static void Config(IUnityContainer unityContainer)
{
unityContainer.RegisterType<ICustomerDataAccess, CustomerSqlDataAccess>();
unityContainer.RegisterType<ICustomerDataAccess, CustomerMysqlDataAccess>("mysql");
unityContainer.RegisterType<Customer>(
new InjectionConstructor(new ResolvedParameter<ICustomerDataAccess>()));
unityContainer.RegisterType<Customer>("mysqlCustomer",
new InjectionConstructor(new ResolvedParameter<ICustomerDataAccess>("mysql")));
}
} public class Customer
{
private ICustomerDataAccess CustomerDataAccess { get; set;} public string Id { get; set; }
public string Name { get; set; } public Customer(ICustomerDataAccess customerDataAccess)
{
CustomerDataAccess = customerDataAccess;
}
public void Save()
{
CustomerDataAccess.Save(this);
}
} class Program
{
static void Main(string[] args)
{
var container = new UnityContainer();
UnitySetup.Config(container);
var sqlCustomer = container.Resolve<Customer>();
var myqlCustomer = container.Resolve<Customer>("mysqlCustomer"); sqlCustomer.Save();
myqlCustomer.Save(); Console.ReadKey();
}
}
}
三 、 方法调用注入(Method Call Injection)
方法调用注入和输入注入有点类似,只需要在调用的方法上面添加[InjectionMethod] 标注即可
1. 配置unity
unityContainer.RegisterType<Customer>("mysqlCustomer",
new InjectionMethod("SetDataAccess", new ResolvedParameter<ICustomerDataAccess>("mysql")));
2. 获取
var sqlCustomer = container.Resolve<Customer>();
var myqlCustomer = container.Resolve<Customer>("mysqlCustomer");
完成后的代码如下
using System;
using System.Collections.Generic;
using System.Data;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Microsoft.Practices.Unity; namespace UnitySample
{ public interface ICustomerDataAccess
{
void Save(Customer c);
} public class CustomerSqlDataAccess : ICustomerDataAccess
{
public void Save(Customer c)
{
Console.WriteLine("{2}, save data id:{0},name{1}",c.Id,c.Name,this.GetType().ToString());
}
} public class CustomerMysqlDataAccess : ICustomerDataAccess
{
public void Save(Customer c)
{
Console.WriteLine("{2}, save data id:{0},name{1}", c.Id, c.Name, this.GetType().ToString());
}
} public class UnitySetup
{
public static void Config(IUnityContainer unityContainer)
{
unityContainer.RegisterType<ICustomerDataAccess, CustomerSqlDataAccess>();
unityContainer.RegisterType<ICustomerDataAccess, CustomerMysqlDataAccess>("mysql"); unityContainer.RegisterType<Customer>("mysqlCustomer",
new InjectionMethod("SetDataAccess", new ResolvedParameter<ICustomerDataAccess>("mysql")));
}
} public class Customer
{
private ICustomerDataAccess CustomerDataAccess { get; set;}
public string Id { get; set; }
public string Name { get; set; } public void Save()
{
CustomerDataAccess.Save(this);
} [InjectionMethod]
public void SetDataAccess(ICustomerDataAccess dataAccess)
{
CustomerDataAccess = dataAccess;
}
} class Program
{
static void Main(string[] args)
{
var container = new UnityContainer();
UnitySetup.Config(container);
var sqlCustomer = container.Resolve<Customer>();
var mysqlCustomer = container.Resolve<Customer>("mysqlCustomer");
sqlCustomer.Save();
mysqlCustomer.Save();
Console.ReadKey();
}
}
}
好了,通过使用unity 依赖注入,customer 对象不再依赖具体的CustomerSqlDataAccess 和 CustomerMysqlDataAccess., 只依赖于接口ICustomerDataAccess。
上面就是unity 的基本用法介绍。
使用Microsoft.Practices.Unity 依赖注入的更多相关文章
- 使用Microsoft.Practices.Unity 依赖注入 转载https://www.cnblogs.com/slardar1978/p/4205394.html
Unity是微软Patterns & Practices团队所开发的一个轻量级的,并且可扩展的依赖注入(Dependency Injection)容器,它支持常用的三种依赖注入方式:构造器注入 ...
- Microsoft.Practices.Unity 给不同的对象注入不同的Logger
场景:我们做项目的时候常常会引用第三方日志框架来帮助我们记录日志,日志组件的用途主要是审计.跟踪.和调试.就说我最常用的日志组件log4net吧,这个在.NET同行当中应该算是用得非常多的一个日志组件 ...
- WPF Microsoft.Practices.Unity 注入大法简单示例
最近新入职了公司,做WPF方向的项目,进来后看到这边大量运用了依赖注入来解耦,采用的是Microsoft.Practices.Unity. WPF的话,目前主要有两个技术来实现IOC,unity和ME ...
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(6)-Unity 依赖注入
系列目录 前言 为了符合后面更新后的重构系统,文章于2016-11-1日重写 本节重构一下代码,采用IOC控制反转,也就是依赖注入 您可以访问http://unity.codeplex.com/rel ...
- Unity依赖注入使用详解
写在前面 构造器注入 Dependency属性注入 InjectionMethod方法注入 非泛型注入 标识键 ContainerControlledLifetimeManager单例 Unity注册 ...
- 利用Microsoft.Practices.Unity的拦截技术,实现.NET中的AOP
1.记住这个单词的意思:Interception(拦截) 2.首先说一下原理和背景 原理:所谓的AOP就是面向切面编程,这里不多说,百度搜索. 目的:个人认为是为了解耦,部分代码跟业务代码分离,业务代 ...
- 第九回 Microsoft.Practices.Unity.Interception实现基于数据集的缓存(针对六,七,八讲的具体概念和配置的解说)
返回目录 概念 Microsoft.Practices.Unity.Interception是一个拦截器,它隶属于Microsoft.Practices.Unity组成之中,主要完成AOP的功能,而实 ...
- Unity 依赖注入知识点
三种依赖注入方法,构造器注入.属性注入.方法注入 可以配置Config文件,来实现不用修改代码.需要先将接口与实体关联,然后使用时会自动加载对应实体. namespace WeChatConsole ...
- Unity依赖注入使用
构造器注入(Constructor Injection):IoC容器会智能地选择选择和调用适合的构造函数以创建依赖的对象.如果被选择的构造函数具有相应的参数,IoC容器在调用构造函数之前会自定义创建相 ...
随机推荐
- ICPC2019上海区域赛 部分题解(正在更新)
K. Color Graph 题意: 给定一个简单图,点个数<=16,删去部分边后,使得该图中无边数为奇数得环,问剩下的边数最大为多少? 思路: 如果一个图中无奇数边的环,那么这个图一定是个二分 ...
- 机器学习实战笔记-5-Logistic回归
Logistic回归 优缺点 适用范围 优点:计算代价不高,易于理解和实现. 缺点:容易欠拟合,分类精度可能不高. 适用于:数值型和标称型数据. 仅用于二分类 原理: 每个特征都乘以一个回归系数> ...
- 关于this、Echarts中的data
this是指当前对象 移除class的jQuery代码:$('ur.nav li:eq(0)').removeClass('active') 添加class的jQuery代码:$('ur.nav li ...
- 【报错】An error happened during template parsing (template: "class path resource [templates/hello1.html]")
页面显示: Whitelabel Error Page This application has no explicit mapping for /error, so you are seeing t ...
- Spring事务传播及数据库事务操作
从Spring 事务配置说起 先看看Spring 事务的基础配置 <aop:aspectj-autoproxy proxy-target-class="true"/> ...
- Quartz-第一篇 认识Quartz
1.什么是Quartz Quartz是一个任务调度框架,借助Cron表达式,Quartz可以支持各种复杂的任务调度.JDK中也提供了简单的任务调度,java.util.Timer. Quartz的三大 ...
- luoguP1312 Mayan游戏 题解(NOIP2011)
luoguP1312 Mayan游戏 题目 #include<bits/stdc++.h> #define ll long long #define rg register #define ...
- 7、numpy——广播
1.广播的引出 广播(Broadcast)是 numpy 对不同形状(shape)的数组进行数值计算的方式, 对数组的算术运算通常在相应的元素上进行. 如果两个数组 a 和 b 形状相同,即满足 a. ...
- hdu 1828 Picture(线段树轮廓线)
Picture Time Limit: 6000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Sub ...
- K The Right-angled Triangles
链接:https://ac.nowcoder.com/acm/contest/338/K来源:牛客网 题目描述 Consider the right-angled triangles with sid ...