多数用户都会将容器中的大部分对象布署为singleton模式。当一个singleton对象需要和另一个singleton对象协作,或者一个非singleton对象需要和另一个非singleson对象协作时,Spring.net都能很好的处理它们的依赖关系。但是,如果对象的生存周期不同,就可能会产生问题。例如,假设一个singleton对象A要使用一个非singleton(原型)对象B,A中的每个方法都会用到B的新实例。由于 A是singleton对象,容器只有会创建它一次,也就是说只有一次给A的属性赋值的机会,所以不可能在每次A需要的时候都给它注入一个新的B。
有一种解决的办法有点违背控制反转原则:类A可以通过实现IObjectFactoryAware接口来获取容器的引用,并调用GetObject("B")在每次需要的时候从容器中请求一个(新的)对象B。但这并不是一个很好的解决方案,因为客户代码此时必须要和Spring.NET发生紧耦合。
通过方法注入,我们可以用更优雅的方式解决类似的问题。(摘自Spring.NET中文手册)
一、查询方法注入
Spring.NET可以动态覆盖对象的抽象方法或虚方法,并且可以在容器内查找已命名对象,查询方法注入就利用了这些功能。个人感觉查询方法注入类似抽象工厂,为之不同的是,可以不用写抽象的实现代码,通过配置文件动态的切换组件。
在lookup-method节点配置name和object属性
实现代码如下:
<!--查询方法-->
<object id="personDao" type="SpringNet_MethodDi.PersonDao, SpringNet_MethodDi" singleton="false"/>
<object id="objectFactory" type="SpringNet_MethodDi.ObjectFactory, SpringNet_MethodDi">
<lookup-method name="CreatePersonDao" object="personDao"/>
</object>
LookupMethod
//注意,可以直接在配置中定义这个类的对象
public abstract class ObjectFactory
{
//或者可以是一个虚方法
public abstract PersonDao CreatePersonDao();
}
public class PersonDao
{
public void Save()
{
Console.WriteLine("保存数据");
}
}
二、替换任意方法
替换任意方法在项目中使用的很少,实现方法也比较复杂。至于Spring.net为什么使用替换任意方法,我还不是很清楚,如果有知道使用场景的朋友可以给我留言。我个人认为其用途是:实现非派生类方法的重写或在IoC框架中简易的AOP拦截(这一点又不确定,以后我会写AOP方面的博客)。
首先继承IMethodReplacer接口并实现Implement方法,object[] arguments为传入的参数。其次在replaced-method节点配置name和replacer属性,和增加arg-type节点且配置返回类型match属性
代码如下:
<!--替换方法-->
<object id="userDao" type="SpringNet_MethodDi.UserDao, SpringNet_MethodDi">
<replaced-method name="GetValue" replacer="replaceValue">
<arg-type match="String"/>
</replaced-method>
</object>
<object id="replaceValue" type="SpringNet_MethodDi.ReplaceValue, SpringNet_MethodDi"/>
ReplacedMethod
public class UserDao
{
//虚方法
public virtual string GetValue(string input)
{
return null;
}
}
//实现IMethodReplacer接口
public class ReplaceValue : IMethodReplacer
{
public object Implement(object target, MethodInfo method, object[] arguments)
{
string value = (string)arguments[0];
return "获取到:" + value;
}
}
三、事件注入
在Spring.net的IoC框架中,除了提供方法注入以外,还提供事件的注入。通过事件的注入,可以使架构体系的耦合降到最低。(参考 clingingboy的 Spring.NET学习笔记(3)-注册事件注入)
在listener节点处配置event和method属性指明事件名和绑定的方法,并增加ref节点设置object属性来指明调用哪个IoC容器对象。
实现代码:
<!--事件注入-->
<object id="men" type="SpringNet_MethodDi.Men, SpringNet_MethodDi">
<listener event="OpenTheDoor" method="OpenThisDoor">
<ref object="door"/>
</listener>
</object>
<object id="door" type="SpringNet_MethodDi.Door, SpringNet_MethodDi" />
Listener
//先定义一个委托
public delegate string OpenHandler(string arg);
public class Door
{
public event OpenHandler OpenTheDoor;
public void OnOpen(string arg)
{
//调用事件
if (OpenTheDoor != null)
{
Console.WriteLine(OpenTheDoor(arg));
}
}
}
public class Men
{
public string OpenThisDoor(string arg)
{
return "参数是:" + arg;
}
}
调用部分代码:
Program
class Program
{
static void Main(string[] args)
{
IApplicationContext ctx = ContextReGIStry.GetContext();
Console.WriteLine("查询方法");
ObjectFactory factory = (ObjectFactory)ctx.GetObject("objectFactory");
factory.CreatePersonDao().Save();
Console.WriteLine();
Console.WriteLine("替换方法");
UserDao dao = (UserDao)ctx.GetObject("userDao");
Console.WriteLine(dao.GetValue("Liu Dong"));
Console.WriteLine();
Console.WriteLine("事件注册");
Door door = (Door)ctx.GetObject("door");
door.OnOpen("Opening!");
Console.WriteLine();
Console.ReadLine();
}
}
输入效果:
Austin Liu 刘恒辉
Project Manager and Software Designer E-Mail:lzhdim@163.com Blog:https://lzhdim.cnblogs.com 欢迎收藏和转载此博客中的博文,但是请注明出处,给笔者一个与大家交流的空间。谢谢大家。 |