谈谈自己了解的spring.NET的依赖注入

     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 +" &nbsp;性别:"+ item.Sex);
            }
            //输出List<string>
            string output=string.Empty;
            foreach (var item in p.WeekWorkDay)
            {
                output += item + " &nbsp;";
            }
            Response.Write("<br/>&nbsp;&nbsp;每周在" + output + "工作");
            output = string.Empty;
            //输出IList
            foreach (var item in p.Hobbys)
            {
                output += item + "&nbsp;";
            }
            Response.Write("<br/>&nbsp;&nbsp;爱好为" + 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

 

源码下载

喜欢就支持下吧,您的支持是我最大的动力!

posted @ 2013-10-28 21:00  flowbywind  阅读(2802)  评论(0编辑  收藏  举报