zoukankan      html  css  js  c++  java
  • 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

  • 相关阅读:
    51nod 1163 最高的奖励
    51nod 1191 消灭兔子
    51nod 2538 二三排列
    node做渲染服务器的实现
    gulp静态服务器的搭建
    canvas中裁切(橡皮檫)的应用--探照灯,点击去除遮罩
    canvas中图片、视频的加载(截图、切片)
    canvas中笔触基础知识
    Gitee的使用流程
    gulp的使用
  • 原文地址:https://www.cnblogs.com/lcplcpjava/p/6747119.html
Copyright © 2011-2022 走看看