• home > java > spring >

    spring框架学习

    Author:zhoulujun@live.cn Date:

    spring是J2EE应用程序框架,是轻量级的IoC和AOP的容器框架,主要是针对javaBean的生命周期进行管理的轻量级容器,可以单独使用,也可以和Struts框架,ibatis框架等组合使用。

    1,什么是spring框架

    springJ2EE应用程序框架,是轻量级的IoCAOP的容器框架,主要是针对javaBean的生命周期进行管理的轻量级容器,可以单独使用,也可以和Struts框架,ibatis框架等组合使用。

     

     

    2,架构概述

     

      1IoC(Inversion of Control)控制反转,对象创建责任的反转,在springBeanFacotoryIoC容器的核心接口,负责实例化,定位,配置应用程序中的对象及建立这些对象间的依赖。XmlBeanFacotory实现BeanFactory接口,通过获取xml配置文件数据,组成应用对象及对象间的依赖关系。

    spring中有三种注入方式,一种是set注入,一种是接口注入,另一种是构造方法注入。

     

        2AOP面向切面编程

       aop就是纵向的编程,如下图所示,业务1和业务2都需要一个共同的操作,与其往每个业务中都添加同样的代码,不如写一遍代码,让两个业务共同使用这段代码。

     

     spring中面向切面变成的实现有两种方式,一种是动态代理,一种是CGLIB,动态代理必须要提供接口,而CGLIB实现是有继承。

     

    3,为什么使用spring框架

      在不使用spring框架之前,我们的service层中要使用dao层的对象,不得不在service层中new一个对象。如下:

     

    1. //dao层对象  
    2. public class UserDao{  
    3.    publicvoid insert(User user){}  
    4. }  
    5.    
    6. //service层对象  
    7. public classUserService{  
    8.    publicvoid insert(User user){  
    9.        UserDaouserdao = new UserDao();  
    10.        userdao.insert(user);  
    11.    }  
    12. }  

     

     

    存在的问题:层与层之间的依赖。

    使用框架后:

     

     

    1. //dao层对象  
    2. public class UserDao{  
    3.     publicvoid insert(User user){}  
    4. }  
    5.    
    6. //service层对象  
    7. public classUserService{  
    8.    privateUserDao userdao;  
    9.    
    10.    publicUserDao getUserdao() {  
    11.       returnuserdao;  
    12.    }  
    13.    publicvoid setUserdao(UserDao userdao) {  
    14.       this.userdao= userdao;  
    15.    }  
    16.    
    17.    publicvoid insert(User user){  
    18.       userdao.insert(user);  
    19.    }  
    20.    
    21. }  

     

    service层要用dao层对象需要配置到xml配置文件中,至于对象是怎么创建的,关系是怎么组合的都交给了spring框架去实现。

     

    4,框架优点

    轻量级的容器框架没有侵入性

    使用IoC容器更加容易组合对象直接间关系,面向接口编程,降低耦合

    Aop可以更加容易的进行功能扩展,遵循ocp开发原则

    创建对象默认是单例的,不需要再使用单例模式进行处理

     

    5,缺点:业务功能依赖spring特有的功能,依赖与spring环境。
     

    spring框架为我们提供了三种注入方式,分别是set注入,构造方法注入,接口注入。接口注入不作要求,下面介绍前两种方式。

     

    1set注入

      采用属性的set方法进行初始化,就成为set注入。

        1)给普通字符类型赋值。

     

    1. public class User{  
    2.    privateString username;  
    3.    
    4.    publicString getUsername() {  
    5.        returnusername;  
    6.    }  
    7.    publicvoid setUsername(String username) {  
    8.       this.username= username;  
    9.    }  
    10. }  

     

     

       我们只需要提供属性的set方法,然后去属性文件中去配置好让框架能够找到applicationContext.xml文件beans标签。标签beans中添加bean标签, 指定idclass值,id值不做要求,class值为对象所在的完整路径。bean标签再添加property 标签,要求,name值与User类中对应的属性名称一致。value值就是我们要给User类中的username属性赋的值。

     

    1. <bean id="userAction"class="com.lsz.spring.action.User" >  
    2. <span style="white-space:pre">  </span><property name="username" value="admin"></property>  
    3. </bean>  

     

     

       2)给对象赋值

     同样提供对象的set方法

     

    1. public class User{  
    2.      private UserService userservice;  
    3.      public UserServicegetUserservice() {  
    4.           returnuser;  
    5.      }  
    6.      public void setUserservice(UserService userservice){  
    7.          this.userservice= userservice;  
    8.      }  
    9. }  

     

     

       配置文件中要增加UserServicebean标签声明及User对象对UserService引用。

     

    1. <!--对象的声明-->  
    2. <bean id="userService" class="com.lsz.spring.service.UserService"></bean>  
    3.    
    4. <bean id="userAction"class="com.lsz.spring.action.User" >  
    5.    <property name="userservice" ref="userService"></property>  
    6. </bean>  

     

      这样配置,框架就会将UserService对象注入到User类中。

     

      3)给list集合赋值

     同样提供set方法

     

    1. public class User{  
    2.     privateList<String> username;  
    3.     publicList<String> getUsername() {  
    4.         returnusername;  
    5.     }  
    6.     publicvoid setUsername(List<String> username) {  
    7.         this.username= username;  
    8.     }  
    9. }  

     

     

     

    1. <bean id="userAction"class="com.lsz.spring.action.User" >  
    2.      <propertynamepropertyname="username">  
    3.            <list>  
    4.                <value>zhang,san</value>  
    5.                <value>lisi</value>  
    6.                <value>wangwu</value>                                  
    7.                </list>  
    8.     </property>  
    9. </bean>  


     

     

     

      4)给属性文件中的字段赋值

     

    1. public class User{  
    2.     privateProperties props ;  
    3.     publicProperties getProps() {  
    4.         returnprops;  
    5.     }  
    6.     publicvoid setProps(Properties props) {  
    7.         this.props= props;  
    8.     }  
    9. }  

     

    1. <bean>  
    2.     <propertynamepropertyname="props">  
    3.         <props>  
    4.            <propkeypropkey="url">jdbc:oracle:thin:@localhost:orl</prop>  
    5.            <propkeypropkey="driverName">oracle.jdbc.driver.OracleDriver</prop>  
    6.            <propkeypropkey="username">scott</prop>  
    7.            <propkeypropkey="password">tiger</prop>  
    8.         </props>  
    9.     </property>  
    10. </bean>  


     

     

    <prop>标签中的key值是.properties属性文件中的名称

     

    注意:

      无论给什么赋值,配置文件中<property>标签的name属性值一定是和对象中名称一致。

     

     

    2构造方法注入

       1)构造方法一个参数

     

    1. public class User{  
    2.     privateString usercode;  
    3.     publicUser(String usercode) {  
    4.         this.usercode=usercode;  
    5.     }  
    6. }  

     

     

     

    1. <bean id="userAction"class="com.lsz.spring.action.User">                          
    2.     <constructor-argvalueconstructor-argvalue="admin"></constructor-arg>                          
    3. </bean>  


     

     

       2)构造函数有两个参数时

      当参数为非字符串类型时,在配置文件中需要制定类型,如果不指定类型一律按照字符串类型赋值。

      当参数类型不一致时,框架是按照字符串的类型进行查找的,因此需要在配置文件中制定是参数的位置

     

     

    1. <constructor-argvalueconstructor-argvalue="admin"index="0"></constructor-arg>                  
    2. <constructor-argvalueconstructor-argvalue="23" type="int"index="1"></constructor-arg>  
    3.    

     

     

     

      这样制定,就是构造函数中,第一个参数为string类型,第二个参数为int类型
     

    单元测试不是头一次听说了,但只是听说从来没有用过。一个模块怎么测试呢,是不是得专门为一单元写一个测试程序,然后将测试单元代码拿过来测试? 我是这么想的。学到spring框架这才知道单元测试原来是这么回事儿。

     

     下面以上一篇文章中set注入的第一个实例为测试对象。进行单元测试。

     

    1,拷贝jar包

     junit-3.8.2.jar(4.x主要增加注解应用)

     

    2,写业务类

    1. public class User{  
    2.     privateString username;  
    3.    
    4.     publicString getUsername() {  
    5.        returnusername;  
    6.     }  
    7.     publicvoid setUsername(String username) {  
    8.        this.username= username;  
    9.     }  
    10.    
    11.     //添加方法  
    12.     publicString login() throws Exception{  
    13.        if("admin".equals(username){  
    14.           return"success";  
    15.        }else{  
    16.           return"error";  
    17.        }  
    18.     }  
    19. }  

     

     

    3,定义测试类

      测试类最好单独建立项目,或者单独定义文件夹存储,需要继承junit.framework.TestCase

     

    4,增加测试方法

      测试方法必须是public,不应该有返回值,方法名必须以test开头,无参数

     测试方法是有执行先后顺序,按照方法的定义先后顺序

     多个测试方法对同一个业务方法进行测试,一般每个逻辑分支结构都有测试到。

     

    1. public class TestUserextends TestCase{   
    2.     publicvoid testUser_Success() throws Exception{   
    3.        //准备数据  
    4.        Useraction = new User();  
    5.        action.setUsername("admin");  
    6.   
    7.        //调用被测试方法  
    8.        Stringresult = action.login();  
    9.    
    10.        //判断测试是否通过  
    11.        assertEquals("success",result);   
    12.     }  
    13. }  

     

     

     

      运行程序,如果测试成功会出现如下图所示的结果

     如果运行失败,有方法没有通过测试,那么就会显示出在哪个方法出错了。上图中绿色的条会变成红色的。

     

    5,测试类的生命周期方法

     

    1.    
    2. //用来进行初始化操作  
    3. @Override  
    4. protectedvoid setUp() throws Exception {  
    5.    System.out.println("setUp...");  
    6. }  
    7.    
    8. //用来做销毁操作  
    9. @Override  
    10. protectedvoid tearDown() throws Exception {  
    11.    System.out.println("tearDown...");  
    12. }  

     

     

    setUp方法会在每一个测试方法前执行一次。tearDown方法会在每一个测试方法后执行一次


    转载本站文章《spring框架学习》,
    请注明出处:https://www.zhoulujun.cn/html/java/spring/2015_0729_207.html