Unity实现依赖注入
定义:Unity Application Block (Unity)是一个轻量级、可扩展的依赖注入容器,所谓依赖注入容器,即组件之间的依赖关系由容器在运行期决定,容器动态的将某种依赖关系注入到组件之 中,实现构建高度松耦合的软件的工具,依赖注入容器处理相互关联组件的所有细节,因此你可以构建一个独立的组件,提升组件重用的概率,为系统搭建一个灵 活、可扩展的平台。
功能:它有助于构建松耦合的应用程序和为开发者提供以下便利:
1,它提供了创建(或者装配)对象实例的机制,简化对象的创建,特别在分层对象结构和依赖的情形下
2,它支持需求的抽象化,这允许开发人员在运行时或在配置文件中指定依赖,简化横切关注点(crosscutting
concerns)的管理
3,它通过把组件配置推给容器来决定,增加了灵活性
4,服务定位能力;这使客户端能够存储或缓存容器
说明:其实说的白一点Unity是一个用来构建插件式系统的解决方案,我们的应用程序有各个功能模块组件组成,各个组 件通过依赖关系与系统高度松耦合。在开发程序时只预留接口,让组件来继承这些接口实现功能,具体怎么实现的应用程序不管,有组件负责,应用程序只管调用接 口,完成功能。
Unity具体实现依赖注入包含构造函数注入、属性注入、方法注入,所谓注入相当赋值,下面一个一个来介绍。
2.1 默认方式
默认方式跟new一个对象,它会根据你提供的材料,选择一个构造函数,即要有构造器要能访问权限,用Public修饰,构造函数的参数也要提供,即IClass也要能解析,不然就报错了,编程注入方式如下:
public static void ConStructorCodeTest1() { IUnityContainer container = new UnityContainer(); //默认注册(无命名),如果后面还有默认注册会覆盖前面的 container.RegisterType<IClass, CbClass>(); container.RegisterType<IStudent, QlinStudent>(); //解析默认对象 IStudent splitClass = container.Resolve<IStudent>(); splitClass.ShowInfo(); }
配置文件方式 如下:
<?xml version="1.0" encoding="utf-8"?> <configuration> <configSections> <section name="unity" type="Microsoft.Practices.Unity.Configuration.UnityConfigurationSection,Microsoft.Practices.Unity.Configuration"/> </configSections> <unity xmlns="http://schemas.microsoft.com/practices/2010/unity"> <!--引用命名空间--> <namespace name="ConsoleApplication1.UnityDemo.Constructor" /> <!--引用程序集--> <assembly name="ConsoleApplication1" /> <!--容器--> <container name="FirstClass"> <!--映射关系--> <register type="IClass" mapTo="CbClass"></register> <register type="IClass" name="ec" mapTo="EcClass"></register> <register type="IStudent" mapTo="QlinStudent"> </register> </container> </unity> </configuration>
以下是加载配置文件
public static void ConStructorConfigTest1() { IUnityContainer container = new UnityContainer(); string configFile = "http://www.cnblogs.com/UnityDemo/Constructor/Unity.config"; var fileMap = new ExeConfigurationFileMap { ExeConfigFilename = configFile }; //从config文件中读取配置信息 Configuration configuration = ConfigurationManager.OpenMappedExeConfiguration(fileMap, ConfigurationUserLevel.None); //获取指定名称的配置节 UnityConfigurationSection section = (UnityConfigurationSection)configuration.GetSection("unity"); //载入名称为FirstClass 的container节点 container.LoadConfiguration(section, "FirstClass"); IStudent splitClass = container.Resolve<IStudent>(); splitClass.ShowInfo(); }
2.2 指定构造函数
如果构造函数有多个,它也会按照上面那样来初始化一个对象,我们还可以显示用InjectionConstructor特性来指定一个构造函数来解析对象,如下声明:
public class QlinStudent : IStudent { private string Name { get; set; } private IClass ToClass { get; set; } public QlinStudent() { } [InjectionConstructor] public QlinStudent(IClass _class,string name) { ToClass = _class; Name = name; } public void ShowInfo() { Console.WriteLine("{0}就读班级:{1}", Name, ToClass.ClassName); } }
2.3 指定参数依赖的注册名称
构造函数中IClass参数,如果IUnityContainer注册了多个,默认是使用无名称的那个注册,也可以通过Dependency依赖哪个名称来指定哪个来注册,代码,指定ec名称如下:
[InjectionConstructor] public QlinStudent([Dependency("ec")]IClass _class) { ToClass = _class; }
下面注册一个名称为ec的映射,如果没有名称ec的映射将报错
public static void ConStructorCodeTest1() { IUnityContainer container = new UnityContainer(); //默认注册(无命名),如果后面还有默认注册会覆盖前面的 container.RegisterType<IClass, CbClass>(); //命名注册 container.RegisterType<IClass, EcClass>("ec"); container.RegisterType<IStudent, QlinStudent>(); //解析默认对象 IStudent splitClass = container.Resolve<IStudent>(); splitClass.ShowInfo(); }
配置文件方式,代码不变,配置中添加一个 name属性就行,如下:
<container name="FirstClass"> <!--映射关系--> <register type="IClass" mapTo="CbClass"></register> <register type="IClass" name="ec" mapTo="EcClass"></register> <register type="IStudent" mapTo="QlinStudent"> </register> </container>
2.4 指定参数值
构造器中的参数也可以依赖一个指定的类型值,如下代码依赖于EcClass类型,可以让构造函数中可以传入一个具体的类型,这也是构造函数传参数,如下:
public static void ConStructorCodeTest1() { IUnityContainer container = new UnityContainer(); //默认注册(无命名),如果后面还有默认注册会覆盖前面的 container.RegisterType<IClass, CbClass>(); //命名注册 container.RegisterType<IClass, EcClass>("ec"); container.RegisterType<IStudent, QlinStudent>(new InjectionConstructor(new CbClass())); IStudent splitClass = container.Resolve<IStudent>(); splitClass.ShowInfo(); }
或者注册一个实例对象,如下:
public static void ConStructorCodeTest1() { IUnityContainer container = new UnityContainer(); IClass cbClass = new CbClass { ClassName="计科051班" }; //实例注册命名实例 container.RegisterInstance<IClass>("ec", cbClass); container.RegisterType<IStudent, QlinStudent>(); IStudent splitClass = container.Resolve<IStudent>(); splitClass.ShowInfo(); }
配置文件也可以指定类型依赖,如下,指定EcClass:
<register type="IStudent" mapTo="QlinStudent"> <constructor> <param name="_class" type="IClass"> <dependency type="EcClass"/> </param> </constructor> </register>
上面已经介绍了传参数,是用InjectionConstructor类型,现在构造函数,多一个参数,如下:
[InjectionConstructor] public QlinStudent([Dependency("ec")]IClass _class, string name) { ToClass = _class; Name = name; }
多了一个name参数,那必须为容器IUnityContainer提供这个参数,没有这个原材料,它无法构造,就会报错,如下代码:
public static void ConStructorCodeTest1() { IUnityContainer container = new UnityContainer(); container.RegisterType<IStudent, QlinStudent>(new InjectionConstructor(new CbClass() { ClassName = "计科051" }, "Qlin")); IStudent splitClass = container.Resolve<IStudent>(); splitClass.ShowInfo(); }
注入参数后,也可以下次解析的时候,通过ParameterOverrides类来覆盖原来的参数,改变参数值,如下:
public static void ConStructorCodeTest1() { IUnityContainer container = new UnityContainer(); container.RegisterType<IStudent, QlinStudent>(new InjectionConstructor(new CbClass() { ClassName = "计科051" }, "Qlin")); IStudent student = container.Resolve<IStudent>(); student.ShowInfo(); //覆盖参数解析 IStudent student1 = container.Resolve<IStudent>(new ParameterOverrides() { {"_class",new EcClass(){ ClassName="电商051"}}, {"name","linq"} }); student1.ShowInfo(); }
3.属性注入
就是Unity容器解析对象时,为属性赋值,有操作权限要Public修饰属性。属性注入方式和构造函数注入类似,只需在需要注入的属性上增加一个 Dependency特性,Dependency指定一个注册名称name参数用来指定注入对象的名称,属性注入也是伴随着类型初始化时注入的,在解析时 自动注入,所以解析时跟以前一样。代码修改如下,在ToClass属性上增加了Dependency特性,来表示这个属性需要注入:
public class QlinStudent : IStudent { public string Name { get; set; } [Dependency("ec")] public IClass ToClass { get; set; } public void ShowInfo() { Console.WriteLine("{0}就读班级:{1}", Name, ToClass.ClassName); } }
代码方式如下:
IUnityContainer container = new UnityContainer(); container.RegisterType<IClass, EcClass>("ec"); container.RegisterType<IStudent, QlinStudent>(); IStudent splitClass = container.Resolve<IStudent>(); splitClass.ShowInfo();
配置文件方式,依赖的<dependency name="ec1" name值 可指定注册时注册的名称:
<unity xmlns="http://schemas.microsoft.com/practices/2010/unity"> <!--引用命名空间--> <namespace name="ConsoleApplication1.UnityDemo.Constructor4" /> <!--引用程序集--> <assembly name="ConsoleApplication1" /> <!--容器--> <container name="FirstClass"> <!--映射关系--> <register type="IClass" mapTo="CbClass"> </register> <register type="IClass" name="ec1" mapTo="EcClass"> <property name="ClassName" propertyType="System.String" value="电商051" /> </register> <register type="IStudent" mapTo="QlinStudent"> <property name="ToClass"> <dependency name="ec1" type="EcClass"/> </property> </register> </container> </unity>
调用效果图:
4.方法注入
用public修饰方法,方法注入也是跟构造函数类似代码修改如下
public class QlinStudent : IStudent { public string Name { get; set; } private IClass ToClass { get; set; } [InjectionMethod] public void InitClass(IClass _class) { ToClass = _class; } public void ShowInfo() { Console.WriteLine("{0}就读班级:{1}", Name, ToClass.ClassName); } }
编程方式注入不变,就是初始化时,注入值,如下:
IUnityContainer container = new UnityContainer(); container.RegisterType<IClass, EcClass>(); container.RegisterType<IStudent, QlinStudent>(); IStudent student = container.Resolve<IStudent>(); student.ShowInfo();
配置文件方式:
<unity xmlns="http://schemas.microsoft.com/practices/2010/unity"> <!--引用命名空间--> <namespace name="ConsoleApplication1.UnityDemo.Constructor5" /> <!--引用程序集--> <assembly name="ConsoleApplication1" /> <!--容器--> <container name="FirstClass"> <!--映射关系--> <register type="IClass" mapTo="CbClass"> </register> <register type="IClass" name="ec1" mapTo="EcClass"> <property name="ClassName" propertyType="System.String" value="电商051" /> </register> <register type="IStudent" mapTo="QlinStudent"> <property name="Name" propertyType="System.String" value="Qlin" /> <method name="InitClass"> <param name="_class" type="IClass"> <dependency name="ec1" type="EcClass"/> </param> </method> </register> </container> </unity>
本文来自博客园,作者:NLazyo,转载请注明原文链接:https://www.cnblogs.com/bile/articles/3068637.html
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 25岁的心里话
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现