Chinaunix首页 | 论坛 | 博客
  • 博客访问: 7887954
  • 博文数量: 701
  • 博客积分: 2150
  • 博客等级: 上尉
  • 技术积分: 13233
  • 用 户 组: 普通用户
  • 注册时间: 2011-06-29 16:28
个人简介

天行健,君子以自强不息!

文章分类

全部博文(701)

文章存档

2019年(2)

2018年(12)

2017年(76)

2016年(120)

2015年(178)

2014年(129)

2013年(123)

2012年(61)

分类: Java

2015-07-14 18:44:30

Spring框架的核心就是控制反转(Inversion of Control)和依赖注入(Dependency Injection),
通过这两方面来实现松耦合。


一、控制反转(Inversion of Control)和依赖注入
使用IoC,对象是被动的接受依赖类,而不是自己主动的去找。
容器在实例化的时候主动将它的依赖类注入给它。
可以这样理解:控制反转将类的主动权转移到接口上,
依赖注入通过xml配置文件在类实例化时将其依赖类注入。


通过下面的实例来逐步的理解:
首先假设有一个需求,类Business需要调用类Dependency的方法f(),
按照日常的做法,得到下面的代码:


//**类Dependency**
public class Dependency {
    public void f() {};
}


//**类Business**
public  class Business {
    Dependency d;
    public Business() {
      d = new Dependency();
    }
    public void doSth() {
        d.f();
    }
}


对上述实现做出如下修改:
首先,将Business里的Dependency实例的获得该为setter方式,
其次,将Dependency类改为某个接口的实现。
故可以得到下面新的代码:


//**接口IDependency**
public interface IDependency {
    void f();
}


//**类Dependency**
public class Dependency {
    public void f() {};
}


//**类Business**
public  class Business {
    IDependency d;
    public Business() {}   // 不生成实例


    public void doSth() {
        d.f();
    }


    public void setDependency(IDependency d) {
        this.d = d;
    }
}


在新的代码中,
首先,Business的变量d可以接收任何IDependency的实例,
另外,Dependency的实例不是通过Business来获得,而是通过setter(也可以用构造器)来由外部传给它。
这似乎跟我们往常的代码没什么不同,但这已经是一个良好的设计。
关键就是Dependency的实例如何从外部注入给Business呢?
这就要通过xml来实现了。


创建一个SpringFirst.xml,进行简单的配置:
<beans>
    <bean id = "dependency" class = "aopfirst.business.Dependency" />
    <bean id = "business" class = "aopfirst.business.Business">
        <property name = "dependency">
            <ref bean = "dependency" />
        </property>
    </bean>
</beans>


这个配置文件里将Dependency类和Business类加入,并将Dependency作为Business的一个参数。
单有了这个xml文件还不够,还需要一个测试类来加载该xml文件,
spring提供了现成的API,在加载上面的xml的时候,
就进行了如下工作:
 . 实例化Dependency类,实例化Business类,
 . 并将Dependency的实例作为参数赋给了Business实例的setDependency()方法。


下面是该测试程序:
public class StartServer {
    public static void main(String [] args) {
        ClassPathResource cr = new ClassPathResource("SpringFirst.xml");
        BeanFactory factory = new XmlBeanFactory(cr);
        Business b = (Business)factory.getBean("business");
        b.doSth();
    }
}




上面的程序加载了xml以后,获得id为"business"的bean,即Business类的实例,并调用了其doSth()方法。
由此可见,Business的依赖类Dependency是通过xml来注入的,而且Business是通过接口IDependency来接收Dependency实例。
因此,当我们又有新的IDependency的实现时,只需要修改xml文件即可,测试程序只需要根据xml里的id值来获得需要的参数。


总结上面的例子,对控制反转和依赖注入已经能理解了。
  . 依赖类(Dependency)是通过外部(xml)来注入的,而不是由使用它的类(Business)来自己制造,这就是依赖的注入。
  . 另一方面,Business对类Dependency的依赖转移到对接口IDependency的依赖,控制权由类转移到了接口,
    即由"实现"转移到"抽象"中。这就是控制反转。


二、Spring中bean的基本xml配置
在spring容器内拼凑bean叫作装配。
装配bean的时候,你是在告诉容器,需要哪些bean,以及容器如何使用依赖注入将它们配合在一起。


理论上,bean装配可以从任何资源获得,包括属性文件,关系数据库等,
但xml是最常见的spring 应用系统配置源。
Spring中的几种容器都支持使用xml装配bean,包括:
  . XmlBeanFactory ,
  . ClassPathXmlApplicationContext ,
  . FileSystemXmlApplicationContext ,
  . XmlWebApplicationContext


基本的xml配置包括如下几个方面:
1.添加一个bean
2.设置bean的属性
2.1 手动设置
2.1.1 通过Setter方法
2.1.2 通过构造器
2.2 自动设置
其中bean的属性即为bean里的成员变量,这些成员变量值的获得可以通过setter方法,
例如某个属性为name,则setter方法为setName(String name);
或者通过构造器在类被实例化时初始化。
Setter方法(例如setName方法)或者构造器的调用都可以通过在xml文件里进行配置,
从而实现让spring容器来自动进行。


1.添加一个bean
以下是一个例子:
    <bean 
        id = “mybean”
        Class = “blog.spring.MyBean”
        Singleton = “false”
        init-method = “initMethod”
        destroy-method = “destroyMethod”
        autowire = “autowire type”
    />


下面是对该标签里各个属性的解释:
  . id :      标识该bean的名称,通过factory.getBean(“id”)来获得实例。
  . class :   该bean的类路径。
  . singleton : 默认为true,即单实例模式,
    每次getBean(“id”)时获取的都是同一个实例,
    如果设置为false,即原型模式,则每次获取的是新创建的实例。
  . init-method : 在bean实例化后要调用的方法(bean里定义好的方法)。
  . Destroy-method : bean从容器里删除之前要调用的方法。
  . Autowire : 其属性要通过何种方法进行属性的自动装配。
    对于上述的各个属性,id和class是必要的,其他的则可以省略。
    例如如果设置了autowire的值,则表明需要自动装配,否则是手动装配。


2.通过Setter方法手动设置bean里的属性
Bean里的属性通过<property>标签来标识。有以下几种情况:
  . 简单类型属性
        <bean id = “mybean” class = “blog.spring.MyBean”>
            <property name = “name”>
                <value>springTest</value>
            </property>
        </bean>
  . 引用其他bean
        <bean id = “mybean” class = “blog.spring.MyBean” />
        <bean id = “mybean1” class = “blog.spring.MyBean1”>
            <property name = “name”>
                <ref bean = “mybean” />
            </property>
        </bean>
也可以将<ref>改为
    <bean class = “..”>
这样叫做内部bean,缺点是无法在其他地方重用这个bean的实例。
  . 装配集合
    共有以下几种集合的装配:
    ****装配List和数组****
        <property name = ”nameList”>
            <list>
                <value>something</value>
                <ref bean = “blog.spring.MyBean” />
                <value>otherThing</value>
            </list>
        </property>
    ****装配Set****
        <property name = ”nameList”>
            <set>
                <value>something</value>
                <ref bean = “blog.spring.MyBean” />
                <value>otherThing</value>
            </set>
        </property>
    ****装配Map****
        <property name = ”nameList”>
            <map>
                <entry key = “key1”>
                    <value>value1</value>
                </entry>
                <entry key = “key2”>
                    <ref bean = “mybean” />
                </entry>
            </map>
        </property>
    ****装配Properties****
        <property name = ”nameList”>
            <props>
                <prop key = “prop1”>value1</prop>
                <prop key = “prop2”>value2</prop>
            </props>
        </property>
  . 设置null
    要将一个属性null,需要通过<null />标签,如果不设置,则属性为默认值(在实例化时)而不是null。
        <property name=”name”> <null /> </property>


3.通过构造器手动设置bean里的属性
假设有如下一个bean:
    Public class MyBean {
        Public MyBean( String arg1, MyBean1 arg2, String arg3 )
    }


则可以在xml里这样配置该bean:
<bean id = “mybean” class = “blog.spring.MyBean”>
        <constructor-arg index = “1”>
            <value>springTest</value>
        <constructor-arg>
        <constructor-arg index = “0”>
            <ref bean = “mybean1” />
        <constructor-arg>
</bean>
其中的index是用来标识该参数在构造函数里的位置的,并从0开始。


4.让spring完成自动装配
例如:
<bean 
id = “mybean” 
class = “blog.spring.MyBean”
autowire = “autowire type”
/>


下面是几种autowire type的说明:
  . byname : 试图在容器中寻找和需要自动装配的属性名相同的bean或id,如果没有找到相应的bean,则这个属性未被装配上。
  . byType : 试图在容器中寻找一个与需要自动装配的属性类型相同的bean或id,如果没有找到,则该属性未被装配上。
  . constructor : 试图在容器中寻找与需要自动装配的bean的构造函数参数一致的一个或多个bean,如果没找到则抛出异常。
  . autodetect : 首先尝试使用constructor来自动装配,然后再使用byType方式。


从上面可以看出,如果某个bean不手动设置autowire属性,则默认为手动装配。
如果需要将所有bean都设置为自动装配时,可以通过在<beans>标签中设置default-autowire属性。
<beans>标签是整个xml文档的根,在它下面就是一个个的<bean>。
其中default-autowire的值也有byName,byType,constructor,autodetect四种。
例如配置如下:
<beans default-autowire = “byName”>
    ...
</beans>


自动装配可能带来不确定性问题。
例如使用byType时可能同时发现两个相同的类型,则不知道该采用哪一个。
所以可能混合采用自动和手动装配。
例如,对某个bean设置为自动装配,而对其某个属性则手动明确的设置其值,
例如:
<bean id = “mybean” class = “blog.spring.MyBean”
    Autowire = “byType”
>
    <property name = “name”>
        <ref bean = “myBean1”>
    </property>
</bean>
通过这样的配置,对mybean里的name属性进行手动装配,而对除name外的其他属性就进行自动装配。
阅读(1332) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~