论坛首页 Java企业应用论坛

关于spring声明式事务管理异常处理的测试和小结

浏览 23510 次
该帖已经被评为良好帖
作者 正文
   发表时间:2006-11-18  
关于spring事务管理以及异常处理的帖子,本论坛争论颇多,各有各的测试代码,也各有各的测试结果,
不知道是spring版本的不同还是各测试的例子的不同而导致测试结果出现差异.
本人也很想弄清楚spring是如何对Service进行事务管理的,并且还去看了一下spring框架关于事务管理几个相关类的源码,可惜由于本人功力有限,只看懂了皮毛.
既然源代码看不懂,那么只有运用例子进行测试,虽然笨了点,不过管是白猫还是黑猫,能捉老鼠就是好猫.:)
为引起不必要的争论,本帖子只针对本案例的测试结果进行小结,并保证此测试代码在本人的运行环境绝对正确.

开发环境:
OS:windows 2003 Server
Web Server: jakarta-tomcat-5.0.28
DataBase Server: MS SQL Server 2000 (打了SP3补丁)
IDE:  Eclipse 3.2.0+MyEclipse 5.0GA

测试案例系统结构:
web层<---->Service层<---->DAO层

web层使用struts 1.1,DAO使用的spring的JDBC,spring版本1.2

数据库中有两张表:
student1和Student2,表结构相同:id,name,address.其中id为主键且为自增长型.
student1表中有一条记录:
id  name       address
1   xiaoming    wuhan

student2表中记录为空


测试情形一:
web层捕获异常并处理,DAO层不捕获异常,Service也不捕获异常.

Service层接口:

public interface StudentManagerService {
  
    public void  bus_method();
}


DAO层接口
public interface StudentDAO {
  
    public void  deleteStudent1();
    public void  insertStudent2();
}


StudentDAO接口的实现:
public class StudentDAOImp extends JdbcDaoSupport implements StudentDAO{
     //删除student1表中的id=1的记录
     public void  deleteStudent1(){
     JdbcTemplate jt=this.getJdbcTemplate();
     jt.update("delete from student1 where id=1");     
   }
     
     //将student1表中删除的记录插入到student2中,但是此方法实现有错,因为
   //id字段设置为自增长的,所以在插入记录时我们不能指定值
      public void  insertStudent2(){
       JdbcTemplate jt=this.getJdbcTemplate();
            String arg[]=new String[3];
	   arg[0]="1";
            arg[1]="xiaoming";
	   arg[2]="wuhan";
            jt.update("insert student2(id,name,address) values(?,?,?)",arg);
     }

} 


StudentManagerService 接口的实现:
public class StudentManagerServiceImp implements StudentManagerService{
  private StudentDAO  stdDAO;
 
  public void setStdDAO(StudentDAO   stdDAO){
     this.stdDAO=stdDAO;
  }
   
  //此方法为事务型的:删除student1中的记录成功且插入student2的记录也成功,
 //如果insertStudent2()方法执行失败,那么deleteStudent1()方法也应该会失败
  public void  bus_method(){
    this.stdDAO.deleteStudent1();
    this.stdDAO.insertStudent2();
  }
  
} 



web层:
三个jsp,一个action:
index.jsp ==>首页面.上面仅仅有一个超链接<a herf="test.do">执行</a>
chenggong.jsp ==>Service执行成功后转向的JSP页面
shibai.jsp ====>Service执行失败后转向的JSP页面

action实现:
public class StudentManagerAction  extends  Action{

     public ActionForward execute(ActionMapping mapping, ActionForm form,
	HttpServletRequest request, HttpServletResponse response) {
         try{
             WebApplicationContext appContext=WebApplicationContextUtils. 
                  getWebApplicationContext(this.getServlet().getServletContext());
	    StudentManagerService stdm=(StudentManagerService)appContext.
                                        getBean("stdServiceManager");
	        stdm.bus_method();
	        return mapping.findForward("chenggong");
	 }
	 catch(DataAccessException e){
		System.err.println("action execute service exception!");
		return mapping.findForward("shibai");
	  }

    }
}



配置文件:

web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="2.4" xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee   http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
  <context-param>
    <param-name>log4jConfigLocation</param-name>
    <param-value>/WEB-INF/log4j.properties</param-value>
  </context-param>
  <context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>/WEB-INF/applicationContext.xml</param-value>
  </context-param>
  <listener>
    <listener-class>org.springframework.web.util.Log4jConfigListener</listener-class>
  </listener>
  <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
  <servlet>
    <servlet-name>action</servlet-name>
    <servlet-class>org.apache.struts.action.ActionServlet</servlet-class>
    <init-param>
      <param-name>config</param-name>
      <param-value>/WEB-INF/struts-config.xml</param-value>
    </init-param>
    <init-param>
      <param-name>debug</param-name>
      <param-value>3</param-value>
    </init-param>
    <init-param>
      <param-name>detail</param-name>
      <param-value>3</param-value>
    </init-param>
    <load-on-startup>0</load-on-startup>
  </servlet>
  <servlet-mapping>
    <servlet-name>action</servlet-name>
    <url-pattern>*.do</url-pattern>
  </servlet-mapping>
</web-app>


sturts-config.xml
<struts-config>
  <action-mappings >
    <action  input="/index.jsp"  path="/test"  type="test.StudentManagerAction   >
      <forward name="chenggong" path="/chenggong.jsp" />
      <forward name="shibai" path="/shibai.jsp" />
    </action>
  </action-mappings>
  <message-resources parameter="test.ApplicationResources" />
</struts-config>



applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "http://www.springframework.org/dtd/spring-beans.dtd">
<beans>
	<bean id="dataSource"
	  class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close" >
	  <property name="driverClassName" value="com.microsoft.jdbc.sqlserver.SQLServerDriver"></property>
	  <property name="url" value="jdbc:microsoft:sqlserver://127.0.0.1:1433;databasename=test"></property>
	   <property name="username" value="sa"></property>
	   <property name="password" value="sa"></property>
	</bean>
	
	 <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
	    <property name="dataSource" ref="dataSource"/>
	 </bean>
		
	<bean id="baseTxProxy" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"  lazy-init="true">
	    <property name="transactionManager">
		<ref bean="transactionManager" />
	     </property>
	     <property name="transactionAttributes">
		 <props>
		     <prop key="*">PROPAGATION_REQUIRED</prop>
		</props>
	     </property>
	</bean>

	<bean id="stdServiceManager"  parent="baseTxProxy" >
	    <property name="target">
	         <bean class="test.StudentManagerServiceImp">    
                       <property name="stdDAO">
                      <ref bean="stdDAO"/>
                       </property>    
                  </bean> 
	    </property> 
	</bean>

	<bean id="stdDAO" class="test.StudentDAOImp">
	   <property name="dataSource" ref="dataSource"/>
	</bean>
</beans>


运行程序:启动服务器,并部署.进入index.jsp页面,点击"执行"超链接"---->页面跳向shibai.jsp
查看控制台:打印有:action execute service exception!
查看数据库: student1表中的[1 xiaoming  wuhan] 记录仍然存在,student2表仍然为空.
小结:如果DAO层和Service不捕获异常而在web层捕获异常,web成功捕获异常,spring事务管理成功!


测试情形二:
web层捕获异常并处理,Service捕获异常并处理,DAO层不捕获异常.

修改StudentManagerServiceImp类
public class StudentManagerServiceImp implements StudentManagerService{
  private StudentDAO  stdDAO;
 
  public void setStdDAO(StudentDAO   stdDAO){
     this.stdDAO=stdDAO;
  }
   
  //此方法为事务型的,删除student1中的记录成功且插入student2的记录也成功
 //如果insertStudent2()方法执行失败,那么deleteStudent1()也应该会失败
  public void  bus_method(){
   try{
      this.stdDAO.deleteStudent1();
      this.stdDAO.insertStudent2();
   }
   catch(DataAccessException de)
       System.err.println("service execute exception!");
    }
  }
  
}


运行程序:启动服务器,并部署.进入index.jsp页面,点击"执行"超链接"---->页面跳向chenggong.jsp
查看控制台:打印有:service execute exception!
查看数据库: student1表中的[1  xiaoming  wuhan] 记录不存在,student2表仍然为空.
小结:如果Service捕获异常并处理而不向外抛出,web层捕获不到异常,spring事务管理失败!


测试情形(还原表中的数据)三:
web层捕获异常,Service捕获异常,DAO层也捕获异常.

修改StudentDAOImp类代码
public class StudentDAOImp extends JdbcDaoSupport implements StudentDAO{
     //删除student1表中的id=1的记录
     public void  deleteStudent1(){
        try{
     JdbcTemplate jt=this.getJdbcTemplate();
     jt.update("delete from student1 where id=1");
       }
       catch(DataAccessException e){
         System.err.println("dao deleteStudent1 execute exception!");
       }     
   }
     
     //将student1表中删除的记录插入到student2中,但是此方法实现有错,因为
   //id字段设置为自增长的,所以在插入记录时我们不能指定值
      public void  insertStudent2(){
          try{
       JdbcTemplate jt=this.getJdbcTemplate();
            String arg[]=new String[3];
	   arg[0]="1";
            arg[1]="xiaoming";
	   arg[2]="wuhan";
            jt.update("insert student2(id,name,address) values(?,?,?)",arg);
         }
         catch(DataAccessException  e){
            System.err.println("dao insertStudent2  execute exception!");

         }
     }

} 


运行程序:启动服务器,并部署.进入index.jsp页面,点击"执行"超链接"---->页面跳向chenggong.jsp
查看控制台:打印有:dao insertStudent2  execute exception!
查看数据库: student1表中的 1,xiaoming,wuhan 记录不存在,student2表仍然为空.
小结如果DAO的每一个方法自己捕获异常并处理而不向外抛出,Service层捕获不到异常,Web层同样捕获不到异常,spring事务管理失败!


测试情形四:

还原数据库中的数据
还原StudentDAOImp类中的方法为测试情形一中的实现
web层捕获异常Service抛出的自定义异常StudentManagerException
Service捕获DataAccessException并抛出StudentManagerException,
StudentManagerException为DataAccessException的子类
DAO层不捕获异常

修改StudentManagerServiceImp类的实现:
public class StudentManagerServiceImp implements StudentManagerService{
  private StudentDAO  stdDAO;
 
  public void setStdDAO(StudentDAO   stdDAO){
     this.stdDAO=stdDAO;
  }
   
  //此方法为事务型的,删除student1中的记录成功且插入student2的记录也成功
 //如果insertStudent2()方法执行失败,那么deleteStudent1()也应该会失败
  public void  bus_method() throws StudentManagerException{
   try{
      this.stdDAO.deleteStudent1();
      this.stdDAO.insertStudent2();
   }
   catch(DataAccessException de)
       System.err.println("service execute exception!");
     throw new StudentManagerException();//StudentManagerException类继承DataAcce                          //ssException异常
    }
  }
}



修改StudentManagerAction 
public class StudentManagerAction  extends  Action{

     public ActionForward execute(ActionMapping mapping, ActionForm form,
	HttpServletRequest request, HttpServletResponse response) {
         try{
             WebApplicationContext appContext=WebApplicationContextUtils. 
                  getWebApplicationContext(this.getServlet().getServletContext());
	    StudentManagerService stdm=(StudentManagerService)appContext.
                                        getBean("stdServiceManager");
	        stdm.bus_method();
	        return mapping.findForward("chenggong");
	 }
	 catch(StudentManagerException e){
		System.err.println("action execute service exception!");
		return mapping.findForward("shibai");
	  }

    }
}

运行程序:启动服务器,并部署.进入index.jsp页面,点击"执行"超链接"---->页面跳向shibai.jsp
查看控制台:打印有:service execute exception!
          action execute service exception!
查看数据库: student1表中的 [1,xiaoming,wuhan] 记录仍然存在,student2表仍然为空.
小结如果DAO的每一个方法不捕获异常,Service层捕获DataAccessException异常并抛出自己定义异常(自定义异常为DataAccessException的子类),Web层可以捕获到异常,spring事务管理成功!



结合源码总结:
1.spring在进行声明时事务管理时,通过捕获Service层方法的DataAccessException来提交和回滚事务的,而Service层方法的DataAccessException又是来自调用DAO层方法所产生的异常.

2.我们一般在写DAO层代码时,如果继承JdbcDaoSupport 类,并使用此类所实现的JdbcTemplate来执行数据库操作,此类会自动把低层的SQLException转化成DataAccessException以及DataAccessException
的子类.

3.一般在Service层我们可以自己捕获DAO方法所产成的DataAccessException,然后再抛出一个业务方法有意义的异常(ps:此异常最好继承DataAccessException),然后在Web层捕获,这样我们就可以手动编码的灵活实现通过业务方法执行的成功和失败来向用户转发不同的页面







   发表时间:2007-01-13  
我测试的结果和你的有点不同,测试的时候发现,即使是空指针一样,一样会回滚事物。
0 请登录后投票
   发表时间:2007-01-15  
yananay 写道
我测试的结果和你的有点不同,测试的时候发现,即使是空指针一样,一样会回滚事物。

你说的空指针是什么情况?能不能给出详细说明,我将继续测试,谢谢!
0 请登录后投票
   发表时间:2007-01-16  
lkf520java 写道
yananay 写道
我测试的结果和你的有点不同,测试的时候发现,即使是空指针一样,一样会回滚事物。

你说的空指针是什么情况?能不能给出详细说明,我将继续测试,谢谢!


例如:

数据库操作1;
空指针;
数据库操作2;

这样, 数据库的更新就会回滚.

其实我也很奇怪为什么会回滚,因为按照spring的说法,只有抛出了 DAOAccessException才会回滚...
0 请登录后投票
   发表时间:2007-01-17  
yananay 写道
lkf520java 写道
yananay 写道
我测试的结果和你的有点不同,测试的时候发现,即使是空指针一样,一样会回滚事物。

你说的空指针是什么情况?能不能给出详细说明,我将继续测试,谢谢!


例如:

数据库操作1;
空指针;
数据库操作2;

这样, 数据库的更新就会回滚.

其实我也很奇怪为什么会回滚,因为按照spring的说法,只有抛出了 DAOAccessException才会回滚...


你的意思是在一事务方法中如下操作流程:

数据库操作1  --->其他操作2,出现空指针异常--->数据库操作3


那么在 数据库操作1 操作无误,其他操作2出现空指针异常的时候,数据库操作1会回滚???
0 请登录后投票
   发表时间:2007-01-17  
yananay 写道
lkf520java 写道
yananay 写道
我测试的结果和你的有点不同,测试的时候发现,即使是空指针一样,一样会回滚事物。

你说的空指针是什么情况?能不能给出详细说明,我将继续测试,谢谢!


例如:

数据库操作1;
空指针;
数据库操作2;

这样, 数据库的更新就会回滚.

其实我也很奇怪为什么会回滚,因为按照spring的说法,只有抛出了 DAOAccessException才会回滚...

你错了!
spring默认对RunTimeException及其子类回滚(例如DAOAccessException),也可以通过配置对任何CheckedException回滚
1 请登录后投票
   发表时间:2007-01-17  
lkf520java 写道
yananay 写道
lkf520java 写道
yananay 写道
我测试的结果和你的有点不同,测试的时候发现,即使是空指针一样,一样会回滚事物。

你说的空指针是什么情况?能不能给出详细说明,我将继续测试,谢谢!


例如:

数据库操作1;
空指针;
数据库操作2;

这样, 数据库的更新就会回滚.

其实我也很奇怪为什么会回滚,因为按照spring的说法,只有抛出了 DAOAccessException才会回滚...


你的意思是在一事务方法中如下操作流程:

数据库操作1  --->其他操作2,出现空指针异常--->数据库操作3


那么在 数据库操作1 操作无误,其他操作2出现空指针异常的时候,数据库操作1会回滚???

可以做到,条件是正确配置了事务传播属性
当“其他操作2”抛出空指针异常时,会导致其所在的事务回滚,如果“数据库操作1”也在此事务上下文内就会跟着一起回滚
0 请登录后投票
   发表时间:2007-01-18  
本人是新手.最近把spring in action 看完.有些不懂 的问题请各位指点下.(由于本人发贴求助.被管理员移到入门讨论中,且也无人回答.万不得已,才来这里打搅大家.)
关于SPRING的事务性操作的.在spring中,我发现
<bean id="transactionAttributeSource" class="org.springframework.transaction.interceptor.NameMatchTransactionAttributeSource">
 <property name="properties">
  <props>
   <prop key="add">
    PROPAGATION_REQUIRES_NEW
   </prop>
  </props>
 </property>
</bean>
中可以通过properties的KEY来匹配我想实现事务操作的方法.
但要是在
<bean id="studentManager" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
 <property name="target">
  <value></value>
 </property>
 <property name="transactionManager">
  <value></value>
 </property>
 <property name="transactionAttributeSource">
  <ref bean="transactionAttributeSource"/>
 </property>
</bean>

<bean id="myTrans" class="org.springframework.transaction.interceptor.DefaultTransactionAttribute">
<property name="propagationBehaviorName">
<value>PROPAGATION_REQUIRES_NEW</value>
</property>
<property name="isolationLevelName">
<value>ISOLATION_REPEATABLE_READ</value>
</property>
</bean>

<bean id="transactionAttributeSource" class="org.springframework.transaction.interceptor.MatchAlwaysTransactionAttributeSource">
<property name="transactionAttribute">
<ref bean="myTrans"/>
</property>
</bean>
中没有指定这个事务所匹配的方法名(或格式)
请赐教.不胜感激
0 请登录后投票
   发表时间:2007-04-28  
请教, 不用web容器 ,直接写测试用例 继承自AbstractTransactionalSpringContextTests时,怎么写测试用例,如何断言?
0 请登录后投票
论坛首页 Java企业应用版

跳转论坛:
Global site tag (gtag.js) - Google Analytics