Spring 学习

一、Spring 简介

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类型

三、junit单元测试


 单元测试不是头一次听说了,但只是听说从来没有用过。一个模块怎么测试呢,是不是得专门为一单元写一个测试程序,然后将测试单元代码拿过来测试? 我是这么想的。学到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方法会在每一个测试方法后执行一次


四、自动装配

  set注入和构造注入有时在做配置时比较麻烦。所以框架为了提高开发效率,提供自动装配功能,简化配置。spring框架式默认不支持自动装配的,要想使用自动装配需要修改spring配置文件中<bean>标签的autowire属性

 自动装配属性有6个值可选,分别代表不同的含义。

 

1,byName

 从Spring环境中获取目标对象时,目标对象中的属性会根据名称在整个Spring环境中查找<bean>标签的id属性值。如果有相同的,那么获取这个对象,实现关联。

 整个Spring环境:表示所有的spring配置文件中查找,那么id不能有重复的。

 

2,byType

 从Spring环境中获取目标对象时,目标对象中的属性会根据类型在整个spring环境中查找<bean>标签的class属性值。如果有相同的,那么获取这个对象,实现关联。

 

 缺点:如果存在多个相同类型的bean对象,会出错。

       如果属性为单一类型的数据,那么查找到多个关联对象会发生错误。

       如果属性为数组或集合(泛型)类型,那么查找到多个关联对象不会发生异常。

 

3,constructor

 使用构造方法完成对象注入,其实也是根据构造方法的参数类型进行对象查找,相当于采用byType的方式。

 

4,autodetect

 自动选择:如果对象没有无参数的构造方法,那么自动选择constructor的自动装配方式进行构造注入。如果对象含有无参数的构造方法,那么自动选择byType的自动装配方式进行setter注入。                      

5,no

 不支持自动装配功能

 

6,default

 表示默认采用上一级标签的自动装配的取值。如果存在多个配置文件的话,那么每一个配置文件的自动装配方式都是独立的。                   

 

 如果配置文件存在多个的情况下,加载配置文件的方式:

 1)可以指定总的配置文件去包含子的配置文件,然后只加载总的配置文件即可在总的配置文件applicationContext.xml中使用import标签进行子文件包<importresource="applicationContext-test.xml"/> 

        

 代码中加载配置文件:   

  1. <span style="font-size:18px;">ApplicationContextacnewClassPathXmlApplicationContext("applicationContext.xml");</span>  

 

 2)使用星号来匹配多个文件进行加载,文件名称要符合规律。 (推荐使用)


  1. <span style="font-size:18px;">   //配置文件的名称  
  2. applicationContext.xml  
  3. applicationContext-action.xml  
  4. applicationContext-service.xml  
  5. applicationContext-dao.xml  
  6.    
  7. ApplicationContextac =newClassPathXmlApplicationContext("applicationContext*.xml");  
  8.  </span>  


 3)可以使用数组作为参数,一次性加载多个配置文件

  1.    
  2. String[]files={"applicationContext.xml","applicationContext-test.xml"};                 
  3. ApplicationContextac = newClassPathXmlApplicationContext(files);   
  4.        


 

 注意:自动装配功能和手动装配要是同时使用,那么自动装配就不起作用。


五、注解

  注解Annotation,是一种类似注释的机制,在代码中添加注解可以在之后某时间使用这些信息。跟注释不同的是,注释是给我们看的,Java虚拟机不会编译,注解也是不编译的,但是我们可以通过反射机制去读取注解中的信息。注解使用关键字@interface,继承java.lang.annotition.Annotition

  

 spring框架为我们提供了注解功能。

  使用注解编程,主要是为了替代xml文件,使开发更加快速。但是,xml文件的使用就是解决修改程序修改源代码,现在又不去使用xml文件,那么不就违背了开闭原则了么,得确是。不过么,注解也有注解的好,使用注解就不用配置那么多的xml文件啦,最重要的是开发效率高。。

  在没有使用注解时,spring框架的配置文件applicationContext.xml文件中需要配置很多的<bean>标签,用来声明类对象。使用注解,则不必在配置文件中添加标签拉,对应的是在对应类的“注释”位置添加说明。具体介绍如下:

 

   spring框架使用的是分层的注解。

    持久层:@Repository

    服务层:@Service

    控制层:@Controller

 

1,使用注解,需要在配置文件中增加命名空间和约束文件

  1. <beans ...  
  2. xmlns:context="http://www.springframework.org/schema/context"  
  3. xsi:schemaLocation="  
  4. ...  
  5. http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context-2.5.xsd  
  6. ">  

  

2,告诉框架那些类是使用注解的。

<context:component-scan base-package="com.lsz.spring" />

 

 

3,持久层注解

  1. Package com.lsz.spring;  
  2.    
  3. @Repository  
  4. public class UserDao{  
  5. //。。。。  
  6. }  

@Repository等同于配置文件中的

  1. <bean id="userDao" class="com.lsz.spring.UserDao" />  


4,服务层注解

  1. @Service(value="testService")  
  2. public classTestService {  
  3.    
  4. @Resource//相当于自动装配  
  5. private UserDao userDao ;  
  6.    
  7.       public UserDao getUserDao() {  
  8.               returnuserDao;  
  9.       }  
  10.       public void setUserDao(UserDao userDao) {  
  11.              this.userDaouserDao;  
  12.       }  
  13.    
  14. }  

@Resource对象间关系的组合,默认采用的是byName方式进行装配,如果根据名称查找不到关联的对象,那么会再采用byType继续查找。

 

@Service注解等同与

  1. <bean id="testService" class="com.lsz.spring.UserService" />  

 

5,控制层注解

  1. @Controller(value="ua")  
  2. @Scope(value="prototype")  
  3. public class UserAction {  
  4.    
  5.      @Resource  
  6.      private UserService userService ;  
  7.    
  8.      public UserService getUserService() {  
  9.           returnuserService;  
  10.      }  
  11. }  

 @Controller注解等同于

  1. <bean id="ua" class="com.lsz.spring.UserAction " />  


 这三个层中的注解关键字都可以使用@Component来代替。

 使用注解声明对象,默认情况下生成的id名称为类名称的首字母小写。

 

6,从Spring环境中获取Action对象。

  1. ServletContext application =request.getSession().getServletContext();  
  2. ApplicationContextac = WebApplicationContextUtils.getWebApplicationContext(application);  
  3.    
  4. UserAction useraction = (UserAction)ac.getBean("ua");//获取控制层对象  
  5.   
  6. response.setContentType("text/html;charset=GBK");//设置编码  
  7. PrintWriter out =response.getWriter();  
  8.   
  9. //分别将三个层的对象打印出来。  
  10. out.println("Action:"+userAction);  
  11. out.println("Service:"+userAction.getUserService());  
  12. out.println("Dao:"+userAction.getUserService().getUserDao()); 

六、AOP

[-]

  1.  AOP的引入
  2. AOP中的术语
  3. spring框架中AOP的使用
    1. 1拷贝jar包
    2. 2spring配置文件中增加命名空间和约束文件
    3. 3写被代理的类和扩展类
    4. 4由配置文件声明
    5. 5测试
  4. Spring框架中使用AOP的优点

AOP(Aspect -OrientedProgramming ) 面向方面编程,与 OOP 完全不同,使用 AOP 编程系统被分为方面或关注点,而不是 OOP 中的对象。

 AOP的引入

 在OOP面向对象的使用中,无可避免的会出现代码重复,而且使用面向对象的编程方式,这种重复无法避免,比如用户权限判断中,根据相应的权限执行相应的方法;在servlet中设置编码格式时,同样相同的代码出现很多次,而且还根业务无关,很容易忘记写,结果运行的时候就出现乱码拉。这种重复代码不仅使编码麻烦,而且还不容易维护。而AOP则将这些代码整理起来,将解决某个切面问题的代码单独的放在某个模块中,然后再织入到程序中。

AOP中的术语




  Aspect:横切面的功能,抽象出类,或接口, AOP编程重要的就是识别出横切面功能。

       方面,类似于字符编码功能) 

 Advice: 横切面功能的具体实现,需要根据实际的情况分析,如果在目标对象操作之前是 before在操作之后,就是 after advice。

       增强,类似于字符编码过滤器

 Pointcut:切入点,描述横切面功能应用的限制,不是所有的流程都需要,那些可以使用的地方就是切入点

       类似于过滤器的匹配规则 /*

 Joinpoint: 连接点,或指组件加入流程的时机,比如设置属性,调用方法,等,spring只支持方法调用的连接点,而其他的一些框架支持属性的连接点如:AspectJ

       类似于过滤器的过滤规则 REQUESTFORWARD

 Weave: 缝合,将组件应用到业务流程中的这个过程,叫缝合或织入。

       类似于将过滤器配置到web.xml文件的过程

 Proxy,代理,在实现上,SpringAOP其实就是使用JDK的动态代理(使用接口的方式完成代理操作),也可以使用CGLIB(使用继承的方式完成代理操作)。

 Target目标,业务操作的实际对象

 

 

实例:设置字符编码格式看作是一个Aspect方面,而拦截器就是一个Advice增强。

  1. <span style="font-size:18px;"><!-- 字符编码过滤器-->  
  2. <filter>  
  3. <filter-name>characterFilter</filter-name>  
  4. <filter-class>com.bjpowernode.egov.filter.CharacterEncodingFilter</filter-class>  
  5. </filter>  
  6. <filter-mapping>  
  7. <filter-name>characterFilter</filter-name>  
  8. <url-pattern>/servlet/*</url-pattern>  
  9. </filter-mapping>  
  10.  </span>  

过滤器

 

  1. <span style="font-size:18px;">public class CharacterEncodingFilter implements Filter {  
  2.    
  3.    @Override  
  4.    public void destroy() {}  
  5.    
  6.    @Override  
  7.    public void doFilter(ServletRequest request, ServletResponse response,  
  8.       FilterChainchain) throws IOException, ServletException {  
  9.            request.setCharacterEncoding("GB18030");  
  10.            chain.doFilter(request,response);  
  11.    }  
  12.    
  13.    @Override  
  14.    publicvoid init(FilterConfig filterConfig) throws ServletException {}  
  15. }</span>  


这样就不用在每个servlet中设置编码拉。。

 

spring框架中AOP的使用

1,拷贝jar



 

2spring配置文件中增加命名空间和约束文件

启用aop功能:就把标签添加进来就可以拉。

 


 

3,写被代理的类和扩展类

 


 


4,由配置文件声明

  1. <span style="font-size:18px;"><!--声明目标类-->  
  2. <bean id="targetclass" class="com.spring.aop.TargetClass"></bean>  
  3.   
  4. <!--声明扩展类-->  
  5. <bean id="extendsclass" class="com.spring.aop.extendsClass"></bean>  
  6.   
  7.   
  8. <!--织入扩展类,并声明在哪个方法上执行扩展类-->  
  9. <aop:config>  
  10.     <aop:aspect id="extendAspect" ref="">  
  11.         <aop:pointcut expression="execution(public ** (..))" id="extendsPoincat">  
  12.         <!--在目标方法执行前执行的方法-->  
  13.         <aop:before method="beforemethod" pointcut-ref="extendsPoincat" />  
  14.         <!--在目标方法执行后执行的方法-->  
  15.         <aop:after method="aftermethod" pointcut-ref="extendsPoincat" />  
  16.     </aop:aspect>  
  17. </aop:config></span>  



 

5,测试

 成功的话,执行目标类中的目标方法targetmethod()时,会先执行扩展类中的beforemethod()方法,再执行目标方法,最后再执行aftermethod()方法。
 也就是我们只需要手动调用targetmethod方法,扩展类中的两个方法框架会在执行的时候通过读取配置文件,获取相应信息,自动给我们添加上扩展的方法。。
 测试肯定成功,不相信的话,可以自己试试。

 

Spring框架中使用AOP的优点

   AopspringIOC容器整合,增强,切入点都是javabean,可以在同一文件中配置

 和spring的其他部分一样,可以在不同应用服务器之间任意移植

   spring实现Aop的拦截接口,使得用户不必绑定在特定的拦截器接口上

 

 aop面向切面的编程思想,打破面向对象的思维方式,我们要学习的不仅是aop的使用,更要学习面向切面的这种思想。



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值