免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1531 | 回复: 0

关于spring声明式事务管理异常处理的测试和小结 [复制链接]

论坛徽章:
0
发表于 2008-12-12 17:41 |显示全部楼层

关于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 ==>首页面.上面仅仅有一个超链接执行
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
  
    log4jConfigLocation
    /WEB-INF/log4j.properties
  
  
    contextConfigLocation
    /WEB-INF/applicationContext.xml
  
  
    org.springframework.web.util.Log4jConfigListener
  
  
    org.springframework.web.context.ContextLoaderListener
  
  
    action
    org.apache.struts.action.ActionServlet
   
      config
      /WEB-INF/struts-config.xml
   
   
      debug
      3
   
   
      detail
      3
   
    0
  
  
    action
    *.do
  
sturts-config.xml
  
   
      
      
   
  
  
applicationContext.xml
       
          
          
          
          
       
       
         
            
         
               
       
            
               
             
             
                 
                     PROPAGATION_REQUIRED
               
             
       
       
            
                     
                       
                     
                           
                  
             
       
       
          
       
运行程序:启动服务器,并部署.进入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层捕获,这样我们就可以手动编码的灵活实现通过业务方法执行的成功和失败来向用户转发不同的页面.

本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u2/88320/showart_1716325.html
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP