spring.net里实现了控制反转IOC(Inversion of control),也即依赖注入DI(Dependency Injection),以达到解耦的目的,实现模块的组件化。程序在调用spring容器的时候,会自动根据配置文件(可以自己命名xml文件,不一定是web.config或app.config)的配置,给你实例化好对象供你调用。这些实例化对象,是基于单例模式的,当然可以在配置文件里修改单例模式singleton为false,这样每次生成的都是在内存中开辟的新的对象。另外也可以在配置文件里设置lazy-init(延迟加载)为true,这样就是只有对象被调用的时候 GetObject(“对象名"),才会实例化对象。
spring.net的依赖注入,支持属性注入,构造器注入,集合注入,方法注入。示例如下:
在相应的xml文件中配置如下:
<objects>
<object id="compute" type="Service.Implement.Compute,Service"></object>
<object id="modernPerson" type="Service.Implement.ModernPerson,Service">
<!--属性注入 Tool对象由compute类注入实现-->
<property name="Tool" ref="compute"></property>
</object>
</objects>
其中modernPerson的结构如下:
public class ModernPerson:IPerson
{
public ITool Tool { get; set; }
public string Work()
{
string str= Tool.UseTool();
return "现代工人: "+str;
}
}
在页面中调用方法为:
IApplicationContext ctx = ContextRegistry.GetContext();
//属性注入 moderPerson的work方法会调用Tool对象的UseTool方法,Compute类实现接口对象Tool
Service.Implement.ModernPerson mp = ctx.GetObject("modernPerson") as Service.Implement.ModernPerson;
Response.Write("<br/>"+mp.Work());
页面输出结果如下:
现代工人: 使用电脑办公:Compute
相应的xml配置文件如下:
<object id="personDao" type="Dao.PersonDao,Dao">
<!--构造器注入 ref为指定的对象 ,若为值类型,则为value-->
<constructor-arg name="per" ref="person"></constructor-arg>
<constructor-arg name="remark" value="welcome"></constructor-arg>
</object>
页面调用方法如下:
//构造器注入
void ConstructorInjection()
{
IApplicationContext ctx = ContextRegistry.GetContext();
PersonDao p = ctx.GetObject("PersonDao") as PersonDao;
Response.Write("<br/>"+ p.ToString());
}
其中personDao重写了tostring方法,以验证是否读取到构造器中的数据,personDao内部代码如下:
private Person _Person;
private string _Remark; public PersonDao(Person per,string remark)
{
_Person = per;
_Remark = remark;
}
public override string ToString()
{
return _Remark + "用户名:" + _Person.Name + " 性别:" + _Person.Sex;
}
页面输入结果如下:
welcome用户名:flowbywind 性别:Boy
关于集合类型,常用的包括List和Dictionary等,下面一一介绍
a)IList类型
使用<list>元素作为IList的标签,element-type属性为泛型的类型,如Int,命名空间.类名,string等;
value为集合中元素的值;
<ref/>表示关联的对象,ref的object属性为关联对象的id或name;
集合也可以为空,用<null/>元素作为标记,此时不声明list元素。
如此时person对象的成员如下:
public IList<Person> Persons { get; set; }
public IList Hobbys;
public IList<string> WeekWorkDay;
其在xml中的配置如下:
<property name="Persons">
<list element-type="Domain.Person,Domain">
<ref object="person"></ref>
</list>
</property>
<property name="WeekWorkDay">
<list element-type="string">
<value>MonDay</value>
<value>Tuesday</value>
<value>Wednesday</value>
<value>Thursday</value>
<value>Friday</value>
</list>
</property>
<property name="Hobbys">
<list>
<value>Reading</value>
<value>Running</value>
</list>
</property>
<!--空集合属性-->
<!--<property name="BestFriends">
<null/>
</property>-->
调用代码为:
//List集合注入
void GenericInjection()
{
IApplicationContext ctx = ContextRegistry.GetContext();
PersonDao p = ctx.GetObject("PersonDao") as PersonDao;
//输出list<Person>对象
foreach (Person item in p.Persons)
{
Response.Write("<br/>采访人:" + item.Name +" 性别:"+ item.Sex);
}
//输出List<string>
string output=string.Empty;
foreach (var item in p.WeekWorkDay)
{
output += item + " ";
}
Response.Write("<br/> 每周在" + output + "工作");
output = string.Empty;
//输出IList
foreach (var item in p.Hobbys)
{
output += item + " ";
}
Response.Write("<br/> 爱好为" + output);
}
最后页面输出结果为:
采访人:flowbywind 性别:Boy
每周在MonDay Tuesday Wednesday Thursday Friday 工作
爱好为Reading Running
b)IDictionary类型
使用<dictionary/>表示IDictionary集合,其中key-type和value-type属性分别为Dict的键值对象类型;
使用<entry>用来表示dict集合的元素,key和value属性为元素的键值队,value-ref为关联的元素
<property name="DicGrade">
<dictionary key-type="string" value-type="object" >
<entry key="2012" value="最佳懒人将"></entry>
<entry key="2013" value-ref="person"></entry>
</dictionary>
</property>
方法注入的目的,是为了解决非单例对象的方法调用。情景为单例对象A,引用了非单例对象B,B此时可能已经注销,但A被生成一次后,无法再次生成,当A想调用B的方法时,就只能重新注入B。A可以通过实现IObjectFactoryAware来获取容器的引用,并通过调用GetObject(“B”)来生成一个新的B对象,但这样做违反了控制反转原则,方法注入此时便是一个比较优雅的解决方案。示例如下:
xml中代码如下:
<object id="objectFactory" type="Dao.ObjectFactory,Dao">
<!--注入查找方法-->
<lookup-method name="CreatePersonDao" object="personDao"></lookup-method>
</object>
其中CreatePersonDao方法是抽象类ObjectFactory的抽象方法,返回PersonDao实例,继而调用personDao里的方法;
相应页面调用的方法为:
Response.Write("<br/>查询方法:");
IApplicationContext ctx = ContextRegistry.GetContext();
ObjectFactory factory = ctx.GetObject("objectFactory") as ObjectFactory;
Response.Write("<br/>"+factory.CreatePersonDao().InjectionFunc());
除了查找方法外,还有替换方法和事件注入,源码中有相应实例;
参考资料:http://tech.ddvip.com/2009-11/1258094512138413_2.html
http://tech.ddvip.com/search.php?key=Spring.NET%E6%95%99%E7%A8%8B
喜欢就支持下吧,您的支持是我最大的动力!