首页 > 代码库 > spring学习总结一----控制反转与依赖注入

spring学习总结一----控制反转与依赖注入

 spring作为java EE中使用最为广泛的框架,它的设计体现了很多设计模式中经典的原则和思想,所以,该框架的各种实现方法非常值得我们去研究,下面先对spring中最为重要的思想之一----控制反转(依赖注入)进行简单的总结。

一、控制反转与依赖注入的概念

  在学习spring框架的时候,我们习惯性地将控制反转和依赖注入放在一起,其实,两者体现的思想原则都是差不多的,只不过控制反转是一种更广泛的程序操作理念,而依赖注入是一种更为具体的实现方法,总的来说,控制反转可以依靠依赖注入来实现。

  1、控制反转的概念。

    在设计模式的总结中,我也曾经在设计模式的博客中总结了控制反转类似的概念思想(可以参考本人这篇博客),它的核心方法理念其实就是通过接口编程,然后让调用者决定实际生产过程用什么类型的接口实现类。当然,在spring中,上面所说的具体类初始化、注入调用代码等过程都是由spring来完成的。上面文字可能稍微抽象化了点,下面上点代码简单说明这个过程,注意看注释:

//应用程序对应的类
public class IocBlog {
    //这里定义接口,通过接口操作对象,从而达到多态的效果
    UserIfc user = null;
    /*spring会通过反射机制,利用setter注入(具体的操作方法是我们在spring配置文件中配置相应bean的property属性即可)
     * 后面的login方法的调用时,就是根据这个注入的类是什么类型进行相应的方法调用的,我们可以针对不同的user实现了注入对应的
     * 用户对象,而应用程序本身不用根据不同的用户种类进行改变,这便达到了代码解耦的目的。当然,注入的过程是由spring帮我们完成的,这个后面再具体讲
     */
    public void setUser(UserIfc user) {
        this.user = user;
    }
    //这是具体的调用方法
    public  void loginDo() {
        //通过接口,引用具体的User对象
        user.login();
    }
}

//先简单定义一个用户接口
interface UserIfc{
    //简单的登录方法
    public void login();
}
//一个具体实现类,该类是在运行时有spring动态地注入的
class User implements UserIfc{
    public void login() {
        System.out.println("user method login");
    }
}

上面这段代码代替说明了控制反转的核心思想--就是通过接口操纵具体实现类,也体现了设计模式中一个很重要的原则:不用程序本身来找具体实现类,而是调用者来决定具体实现类是什么。

下面再具体说明,在spring中如何实现类注入的。 在进行具体的用法之前,有必要先理解依赖注入这个概念。 

  2、依赖注入概念。

  上面也提到,依赖注入是控制反转的一种具体实现手段,通过依赖注入方法可以达到控制反转的目的。其实依赖注入个人理解就是:通过反射机制在程序运行的时候动态地创建具体实现类的对象,然后在需要该对象的时候通过某种手段(一般是通过该bean对象的对应id)将该对象拿出来(也就是注入到具体需要用的地方)。当然,上面这些复制的过程包括类的创建、维护、注入等工作都是spring帮我们完成的。

 

理解了依赖注入和控制反转之后,废话不多说,下面具体看看在spring中如何具体地运用。

 

二、spring中如何注入对象。

  1、具体的注入方法

  接着上面的user类例子说明spring中对象注入的方法,先看下面改进之后的代码(加了个main方法表示具体的调用过程)

//应用程序对应的类
public class IocBlog {
    //这里定义接口,通过接口操作对象,从而达到多态的效果
    UserIfc user = null;
    /*spring会通过反射机制,利用setter注入(具体的操作方法是我们在spring配置文件中配置相应bean的property属性即可)
     * 后面的login方法的调用时,就是根据这个注入的类是什么类型进行相应的方法调用的,我们可以针对不同的user实现了注入对应的
     * 用户对象,而应用程序本身不用根据不同的用户种类进行改变,这便达到了代码解耦的目的。当然,注入的过程是由spring帮我们完成的,这个后面再具体讲
     */
    public void setUser(UserIfc user) {
        this.user = user;
    }
    //这是具体的调用方法
    public  void loginDo() {
        //通过接口,引用具体的User对象
        user.login();
    }
    
    //定义main方法表示具体的调用过程
    public static void main(String [] args){
        //加载配置文件上下文,该对象存储有类之间的依赖关系
        ApplicationContext appContext = new ClassPathXmlApplicationContext("applicationContext.xml"); 
        //通过上下文对象获取具体的bean对象,该对象具体类型在配置文件中指明
        IocBlog test = (IocBlog)appContext.getBean("testId");//一般通过id获取
        //通过loginDo调用login,不同的User类的注入对象会调用不同的对应login方法,他们之间的依赖关系通过配置文件指明
        test.loginDo();
    }
}

//先简单定义一个用户接口
interface UserIfc{
    //简单的登录方法
    public void login();
}
//一个具体实现类,该类是在运行时有spring动态地注入的
class User implements UserIfc{
    private String userName = null;//用户名
    public String getUserName() {
        return userName;
    }
    public void setUserName(String userName) {
        this.userName = userName;
    }

    public void login() {
        System.out.println("user method login");
    }
}

  然后,我们通过以下配置文件进行类注入操作,请注意看注释:

<!-- 这里声明了bean,意思是告诉spring:我有这样一个类,它在 review.blog.springRevice.User路径下
        这个类需要你初始化,因为我在将来的某个时刻可能要用该类(具体初始化可以通过配置来决定顺序,这里不详细讲)
    -->
    <bean id="userId" class="review.blog.springRevice.User">
        <!-- property是对应有setter方法的属性值,必须和name一致 -->
        <property name="userName" value="UserName1"/>
    </bean>
    
    <bean id="testId" class="review.blog.springRevice.IocBlog">
        <!-- 通过ref引用userId,这时,sprign会帮我们将userId对应的对象注入到testId对应的对象的user属性中 -->
        <property name="user" ref="userId"></property>
    </bean>

  可以看到,通过spring,我们可以实现代码组件之间最大程度的松耦合,对象之间的依赖关系不会硬编码到代码中去,而是通过一个配置文件进行映射操作,这大大提高了代码的重用性以及可扩展性。

  2、spring中实现注入原理的简单总结

    总的来说,注入的实现方法还是反射。spring通过dom4j等组件将配置文件加载解析获得对应的类名字之后,利用class.forName方法加载对象。在bean维护中,spring是通过一个map进行对象的维护的,key对应的就是id值,value对应的是对象(从这里看出,spring中bean对象默认是单例的,当然,通过配置文件我们可以改变单例模式,这个不详细讲),具体就不粘贴源码了,有兴趣的可以自己看spring的源码。

  3、其他数据类型的配置使用方法

    在spring中,不仅仅支持普通对象,它也支持map和list等集合类,具体的配置方式也比较简单,下面简单上个demo供参考:

  Map配置方式

<!--map配置方式 -->
<bean id="test" class="Test">   
        <property name="testMap">   
            <map>   
                <entry key="a">   
                    <value>1</value>   
                </entry>   
                <entry key="b">   
                    <value>2</value>   
                </entry>   
            </map>   
        </property>   
    </bean>  

  list配置方式

<!-- list配置方式 -->
    <bean>
        <property name="listTestString">
            <list>
                <value>这里可以是字符串</value>
                <value>也可以不是字符串,具体看下面</value>
            </list>
        </property>
        <property name="listTestRef">
            <list>
                <ref bean="beanId1"/>
                <ref bean="beanId2"/>
            </list>
        </property>
    </bean>

 

  当然,spring还支持很多其他方面的配置,具体就不一一讲解了,可以去官网看spring api。

 

spring学习总结一----控制反转与依赖注入