spring AOP的异常拦截

来源:互联网 发布:淘宝上明星周边 编辑:程序博客网 时间:2024/05/21 16:40

springaopexceptionuserobjectdao

系统的异常处理机制是衡量一个系统设计的关键因素,良好的异常处理机制能在系统出现异常时准确的找到问题的所在。

spring aop对异常的处理有良好的支持。spring 提供了一个接口 ThrowsAdvice,该接口里面没有任何方法,但是实现类里面必须的实现

afterThrowing(Method method, Object[] args, Object target, RuntimeException  throwable) 或者

afterThrowing(RuntimeException  throwable)

如果需要记录发生异常方法的详细信息,则实现第一个方法就行,如果只记录发生的异常,实现第二个方法就ok!

那么异常的处理应该在什么位置来做处理呢?

一般我们的系统都应该有以下几个层次:Action--->Service---->DAO

DAO负责直接和数据库打交道,也是发生异常频率较高的地方,而service只是调用DAO所提供给外面的接口,action里面大部分的操作也是调用service的服务,再加上少数其他的逻辑,这部分的异常可以单独处理!下面我们主要关心DAO层的异常处理。

1、定义接口

[java] view plaincopyprint?
  1. package com.beckham.dao; 
  2. import java.util.List; 
  3. import com.beckham.model.User; 
  4. /**
  5. *  @author Owner
  6. *  Jan 19, 2010   10:15:32 PM
  7. *  struts2
  8. *  com.beckham.dao
  9. *  UserDAO.java
  10. */ 
  11. public interface UserDAO { 
  12.     public boolean userExsit(String username)throws Exception; 
  13.     public User findById(int id)throws Exception; 
  14.     public List<User> queryUser(String hql,int beginIndex)throws Exception; 
  15.     public void saveUser(User user)throws Exception; 
  16.     public  int gettotalSize(String hql)throws Exception ; 

2、实现

[java] view plaincopyprint?
  1. package com.beckham.daoimp; 
  2. import java.util.List; 
  3. import com.beckham.dao.SuperDAO; 
  4. import com.beckham.dao.UserDAO; 
  5. import com.beckham.model.User; 
  6. import com.beckham.util.PropertyUtil; 
  7. /**
  8. *  @author Owner
  9. *  Jan 19, 2010   10:15:50 PM
  10. *  struts2
  11. *  com.beckham.daoimp
  12. *  UserDAOImp.java
  13. */ 
  14. public class UserDAOImpextends SuperDAOimplements UserDAO { 
  15.     public User findById(int id)throws Exception { 
  16.         User user = null
  17.         try
  18.             user = (User) this.getHibernateTemplate().get(User.class, id); 
  19.         } catch (Exception e) { 
  20.             e.printStackTrace(); 
  21.             throw new Exception("主键查询用户失败", e); 
  22.         } 
  23.         return user; 
  24.     } 
  25.     public void saveUser(User user)throws Exception { 
  26.         try
  27.             this.getHibernateTemplate().save(user); 
  28.         } catch (Exception e) { 
  29.             e.printStackTrace(); 
  30.             throw new Exception("增加用户失败", e); 
  31.         } 
  32.     } 
  33.     @SuppressWarnings("unchecked"
  34.     public List<User> queryUser(String hql,int beginIndex)throws Exception { 
  35.         try
  36.             return (List<User>)this.getHibernateTemplate().getSessionFactory() 
  37.                     .getCurrentSession().createQuery(hql).setFirstResult( 
  38.                             beginIndex).setMaxResults( 
  39.                             PropertyUtil.getPageSize()).list(); 
  40.         } catch (Exception e) { 
  41.             e.printStackTrace(); 
  42.             throw new Exception("查询用户出现异常", e); 
  43.         } 
  44.     } 
  45.     public boolean userExsit(String username)throws Exception { 
  46.         boolean bl = true
  47.             String hql = "from User where username='" + username +"'"
  48.                 if (queryUser(hql,0).size() ==0) { 
  49.                     bl = false
  50.                 }    
  51.         return bl; 
  52.     } 
  53.     public int gettotalSize(String hql)throws Exception { 
  54.         int totalsize = 0
  55.         try
  56.             totalsize = Integer.parseInt(this.getHibernateTemplate().find(hql) 
  57.                     .get(0).toString()); 
  58.         } catch (Exception e) { 
  59.             e.printStackTrace(); 
  60.             throw new Exception("查询用户总数失败", e); 
  61.         } 
  62.         return totalsize; 
  63.     } 

这里需要说明的是,这里的异常我没有细致的分类,都是throws Exception。

service层的代码就省略了,因为只是调用DAO层的方法,下面来写异常的拦截

[java] view plaincopyprint?
  1. package com.beckham.aop; 
  2. import java.lang.reflect.Method; 
  3. import org.springframework.aop.ThrowsAdvice; 
  4. /**
  5. * @author Owner Jan 18, 2010 2:37:10 PM 处理DAO层的异常 struts2 com.beckham.aop
  6. *         ExceptionLog.java
  7. */ 
  8. public class ExceptionLogimplements ThrowsAdvice { 
  9.     /**
  10.      * Owner
  11.      * 参数解释 Method method 执行的方法
  12.      * Object[] args 方法参数
  13.      *  Object target 代理的目标对象
  14.      * Throwable throwable 产生的异常
  15.      * Jan 18, 2010 3:21:46 PM
  16.      */ 
  17.     public void afterThrowing(Method method, Object[] args, Object target, 
  18.             RuntimeException  throwable) { 
  19.         System.out.println("产生异常的方法名称:  " + method.getName()); 
  20.          
  21.         for(Object o:args){ 
  22.             System.out.println("方法的参数:   " + o.toString()); 
  23.         } 
  24.          
  25.         System.out.println("代理对象:   " + target.getClass().getName()); 
  26.         System.out.println("抛出的异常:    " + throwable.getMessage()+">>>>>>>" 
  27.                 + throwable.getCause()); 
  28.         System.out.println("异常详细信息:   "+throwable.fillInStackTrace()); 
  29.     } 

最后当然就是在配置文件里面配置了

[xhtml] view plaincopyprint?
  1. <beanid="log"class="com.beckham.aop.LogAdvice"></bean> 
  2.     <bean id="exceptionLog"class="com.beckham.aop.ExceptionLog"></bean> 
  3.     <!-- beanName自动代理 --> 
  4.     <bean id="logAdvice" 
  5.         class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"> 
  6.         <propertyname="beanNames"> 
  7.             <list> 
  8.                 <value>userDAO</value> 
  9.             </list> 
  10.         </property> 
  11.         <propertyname="interceptorNames"> 
  12.         <list> 
  13.             <value>log</value> 
  14.             <value>exceptionLog</value> 
  15.         </list> 
  16.         </property> 
  17.     </bean> 

到此,通过spring AOP拦截异常就完成了,这里只是拦截DAO层的异常,此方法的好处就是处理异常和功能实现完全分离开,只需要在写方法的时候要记得抛出相应的异常,当出现异常时ThrowsAdvice就会拦截到该异常,并获取该异常的详细信息。

 

 

这里贴出我测试可以使用的配置:

<bean id="exceptionLog" class="net.chinanets.utils.AopExceptionHandler"></bean>
    <!-- beanName自动代理 -->
 <bean id="logAdvice"
  class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
  <property name="beanNames">
   <list>
    <value>*Service</value>
   </list>
  </property>
  <property name="interceptorNames">
  <list>
   <value>exceptionLog</value>
  </list>
  </property>
 </bean>


注:

 <property name="beanNames">
   <list>
    <value>*Service</value>
   </list>
  </property>

 

   <value>*Service</value>可以拦截所有的service

*Service匹配你beans.xml中配置的所有bean的ID

这里的切入点是匹配beans ID比较简便,如果从文件路径切入可能会各种报错。

0 1