走自己的路

路漫漫其修远兮,吾将上下而求索

  BlogJava :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 管理 ::
  50 随笔 :: 4 文章 :: 118 评论 :: 0 Trackbacks

应用项目大致的体系结构:

    
 

 该异常处理框架满足的要求:

 

  • 完整的异常组织结构
  • 异常的统一处理
  • 可配置,受管式,方便使用

 

完整的异常组织结构:

  • 用户可以方便的定义自己的异常,但所有UncheckedException需要继承BaseAppRuntimeException,所有的checked Exception可以继承BaseAppException,或者需要抛出且不需要check时用WrapperredAppException封装后抛出
  • 合理地使用checked异常
  • Exception有唯一的error code,这样用户报告异常后,可以根据异常号找到相应Exception,把exception直接显示给用户也没有太大的意义,如何纪录exception那就是下文讲到的ExceptionHandler的职责了。
  • 如果是第三方包括jdk中的异常,需要封装成BaseAppException或者BaseAppRuntimeException后抛出

                                     

   

 

统一的异常处理

异常统一在框架中进行处理,不需要在上层应用的代码中去处理抛出的异常。为了尽量捕捉到所有的异常,将异常处理放在了ActionBroker中,这样凡是action以后抛出的异常都可以捕捉到,因为webservice只是简单的调用action类的方法,一般不会出现异常。当我们捕捉到异常后,需要进行异常处理,定义了ExceptionHandler接口,用接口抽象出异常处理类的具体实现。

 


                        
 

USFContextFactory: 创建ExceptionContext的工厂

 1package com.ldd600.exception.context;
 2
 3public class CoreContextFactory {
 4    private static CoreContextFactory instance;
 5
 6    private volatile ExceptionContext exceptionContext;
 7
 8    private Object exceptionContextLock = new Object();
 9
10    private CoreContextFactory() {
11
12    }

13
14    public static synchronized CoreContextFactory getInstance() {
15        if (null == instance) {
16            instance = new CoreContextFactory();
17        }

18        return instance;
19    }

20
21    public ExceptionContext getExceptionContext() {
22        ExceptionContext tempExpContext = exceptionContext;
23        if (tempExpContext == null
24            synchronized (exceptionContextLock) {
25                tempExpContext = exceptionContext;
26                if (tempExpContext == null)
27                    exceptionContext = tempExpContext = new ExceptionContext();
28            }

29        }

30        return tempExpContext;
31    }

32}

33



   

ExceptionContext: 存放全局的exception信息

 

  1package com.ldd600.exception.context;
  2
  3import java.util.ArrayList;
  4import java.util.Collection;
  5import java.util.Collections;
  6import java.util.HashMap;
  7import java.util.List;
  8import java.util.Map;
  9import java.util.Set;
 10
 11import org.springframework.util.Assert;
 12
 13import com.ldd600.exception.base.BaseAppRuntimeException;
 14import com.ldd600.exception.base.ConfigException;
 15import com.ldd600.exception.base.handler.ExceptionHandler;
 16import com.ldd600.exception.config.ExceptionDefinition;
 17
 18public class ExceptionContext {
 19    private Map<Class<?>, ExceptionDefinition> exceptionMap;
 20
 21    private Map<String, ExceptionHandler> handlers = new HashMap<String, ExceptionHandler>();
 22
 23    ExceptionContext() {
 24        exceptionMap = new HashMap<Class<?>, ExceptionDefinition>();
 25    }

 26
 27    public boolean containsException(Class<?> expClazz) {
 28        return (exceptionMap.containsKey(expClazz));
 29    }

 30    
 31    public void addExceptionHander(Class<?> expClazz, Class<? extends ExceptionHandler> handlerClazz) {
 32        try {
 33            ExceptionDefinition definition = getRealExceptionDefinition(expClazz);
 34            if (null == definition) {
 35                throw new IllegalArgumentException(expClazz.getName() + "not in the context, please configure or add it to the context first!!");
 36            }
 
 37            ExceptionHandler handler = handlers.get(handlerClazz.getName());
 38            if (null == handler) {
 39                handler = handlerClazz.newInstance();
 40                handlers.put(handlerClazz.getName(), handler);
 41            }

 42            
 43            definition.getHandlerNames().add(handlerClazz.getName());
 44        }
 catch (Exception ex) {
 45            throw new ConfigException("Add exception handler to context failure!", ex);
 46        }

 47    }

 48    
 49    public void addExceptionHandler(Class<?> expClazz, String errorCode, Class<? extends ExceptionHandler> handlerClazz) {
 50        Assert.hasLength(errorCode, expClazz + " errorCode must not be null or empty string!");
 51        ExceptionDefinition definition = getRealExceptionDefinition(expClazz);
 52        if(null == definition) {
 53            definition = new ExceptionDefinition(errorCode);
 54            exceptionMap.put(expClazz, definition);
 55        }

 56        addExceptionHander(expClazz, handlerClazz);
 57    }

 58    
 59    
 60    
 61    public void addExceptionHandlers(Class<?> expClazz, Class<? extends ExceptionHandler> handlerClazzes) {
 62        for(Class<? extends ExceptionHandler> handlerClazz : handlerClazzes) {
 63            addExceptionHander(expClazz, handlerClazz);
 64        }

 65    }

 66
 67    public void removeExceptionHandler(Class<?> expClazz, Class<? extends ExceptionHandler> handlerClazz) {
 68        Assert.isTrue(containsException(expClazz));
 69        String handlerName = handlerClazz.getName();
 70        getExceptionDefinition(expClazz).getHandlerNames().remove(handlerName);
 71        Collection<ExceptionDefinition> definitons = exceptionMap.values();
 72        boolean isClearHandler = true;
 73        for (ExceptionDefinition expDefinition : definitons) {
 74            if (expDefinition.getHandlerNames().contains(handlerName)) {
 75                isClearHandler = false;
 76                break;
 77            }

 78        }

 79
 80        if (isClearHandler) {
 81            handlers.remove(handlers.get(handlerName));
 82        }

 83    }

 84
 85    public void setExceptionDefinition(Class<?> expClazz, ExceptionDefinition definition) {
 86        exceptionMap.put(expClazz, definition);
 87    }

 88
 89    public ExceptionDefinition getExceptionDefinition(Class<?> expClazz) {
 90        if (containsException(expClazz)) {
 91            return exceptionMap.get(expClazz);  
 92        }
 else if (BaseAppRuntimeException.class.isAssignableFrom(expClazz.getSuperclass())) {
 93            return getExceptionDefinition(expClazz.getSuperclass());
 94        }
 else {
 95            return null;
 96        }

 97    }

 98    
 99    public ExceptionDefinition getRealExceptionDefinition(Class<?> expClazz) {
100        return exceptionMap.get(expClazz);
101    }

102
103    public List<ExceptionHandler> getExceptionHandlers(Class<?> expClazz){
104        ExceptionDefinition definition = getExceptionDefinition(expClazz);
105        if (null != definition) {
106            Set<String> handlerNames = definition.getHandlerNames();
107            List<ExceptionHandler> handlerList = new ArrayList<ExceptionHandler>(handlerNames.size());
108            for (String handlerName : handlerNames) {
109                ExceptionHandler handler = handlers.get(handlerName);
110                handlerList.add(handler);
111            }

112            List<ExceptionHandler> resultHandlerList = new ArrayList<ExceptionHandler>(handlerList);
113            return resultHandlerList;
114        }
 else {
115            return Collections.<ExceptionHandler> emptyList();
116        }

117    }

118    
119    public String getErrorCode(Class<?> expClazz){
120        ExceptionDefinition definition = getExceptionDefinition(expClazz);
121        if (null != definition) {
122            return definition.getErrorCode();
123        }
 else {
124            return "";
125        }

126    }

127    
128    
129}

130
 

ExceptionDefinition: Exception信息单元

 

 1package com.ldd600.exception.config;
 2
 3import java.util.LinkedHashSet;
 4import java.util.Set;
 5
 6public class ExceptionDefinition {
 7    private String errorCode;
 8
 9    private Set<String> handlerNames = new LinkedHashSet<String> ();
10
11    ExceptionDefinition() {
12        
13    }

14    
15    public ExceptionDefinition(String errorCode) {
16        this.errorCode = errorCode;
17    }

18    
19    public String getErrorCode() {
20        return errorCode;
21    }

22
23    public void setErrorCode(String errorCode) {
24        this.errorCode = errorCode;
25    }

26
27    public Set<String> getHandlerNames() {
28        return handlerNames;
29    }

30}

31
 

ExceptionDefiniton定义了和某个exception相关的具体信息,根据exceptionclass name可以从exceptionContext中的exceptionMap得到指定的exception的相关信息,这些信息是在系统初始化时读取到exceptionContext中的。并且避免了exception handler的重复初始化。

 

可配置,受管式,方便使用

 采取两种配置方式,exception的相关信息比如它的errorCode exceptionHandlers可以配置在外部的xml文件中,也可以用annotation标注。对于exception的处理是有继承性质的,如果某个exception没有在exceptionContext中注册,就使用它的父类的配置信息。如果无任何父类在exceptionContext中注册,就使用默认机制进行处理。

 

XML 方案:

            因为spring2.0支持自定义schema功能,我们可以方便地采用自己的schema只要实现NamespaceHandlerBeanDefinitionPaser,后面一个比较重要,可以将自定义xml文件中的相关类注册到spring的上下文中,成为spring bean

Xml schema:

<xsd:complexType name="exceptionType">
        
<xsd:sequence>
            
<xsd:element name="level" default="error" minOccurs="0">
                
<xsd:simpleType>
                    
<xsd:restriction base="xsd:string">
                        
<xsd:enumeration value="error" />
                        
<xsd:enumeration value="warning" />
                        
<xsd:enumeration value="info" />
                        
<xsd:enumeration value="confirmation" />
                    
</xsd:restriction>
                
</xsd:simpleType>
            
</xsd:element>
            
<xsd:element name="handler" maxOccurs="unbounded">
                
<xsd:simpleType>
                    
<xsd:restriction base="xsd:string" />
                
</xsd:simpleType>
            
</xsd:element>
        
</xsd:sequence>
        
<xsd:attribute name="errorCode">
            
<xsd:simpleType>
                
<xsd:restriction base="xsd:string">
                    
<xsd:whiteSpace value="preserve" />
                    
<xsd:pattern value="LDD600-+\d{1,5}.*" />
                
</xsd:restriction>
            
</xsd:simpleType>
        
</xsd:attribute>
        
<xsd:attribute name="class" type="xsd:string" use="required" />
    
</xsd:complexType>

 

Annotation方案:

            JDK1.5以上就有了annotation,可以简化我们的配置,使得配置信息和代码联系在一起,增加了代码的可读性。如何在spring中注册自定义的annotation和用annotation标注的class,可以参考文章2和文章:   。对于每个注册了的classExceptionalAnnotationBeanPostProcessorparse具体的annotation信息(对于annotationparse方法还会在以后继续改进)。

 1package com.ldd600.exception.annotation;
 2
 3import java.lang.annotation.Documented;
 4import java.lang.annotation.ElementType;
 5import java.lang.annotation.Retention;
 6import java.lang.annotation.RetentionPolicy;
 7import java.lang.annotation.Target;
 8
 9import com.ldd600.exception.base.handler.ExceptionHandler;
10
11@Target({ElementType.TYPE})
12@Retention(RetentionPolicy.RUNTIME)
13@Documented
14public @interface Exceptional {
15    String errorCode();
16    Class<? extends ExceptionHandler>[] handlers();
17}

18


 1package com.ldd600.exception.processor;
 2
 3import org.springframework.beans.BeansException;
 4import org.springframework.beans.factory.config.BeanPostProcessor;
 5
 6import com.ldd600.exception.annotation.Exceptional;
 7import com.ldd600.exception.base.BaseAppException;
 8import com.ldd600.exception.base.BaseAppRuntimeException;
 9import com.ldd600.exception.config.ExceptionDefinition;
10import com.ldd600.exception.context.ExceptionContext;
11import com.ldd600.exception.context.CoreContextFactory;
12
13public class ExceptionalAnnotationBeanPostProcessor implements BeanPostProcessor {
14
15    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
16       if(bean instanceof BaseAppRuntimeException || bean instanceof BaseAppException) {
17           Exceptional exceptional = bean.getClass().getAnnotation(Exceptional.class);
18           if(null != exceptional) {
19               ExceptionContext ctx = CoreContextFactory.getInstance().getExceptionContext();
20               if(!ctx.containsException(bean.getClass())) {
21                   ExceptionDefinition expDefinition = new ExceptionDefinition(exceptional.errorCode());
22                   ctx.setExceptionDefinition(bean.getClass(), expDefinition);
23               }

24               ctx.addExceptionHandlers(bean.getClass(), exceptional.handlers());
25               return null;
26           }

27       }

28       return bean;
29    }

30
31    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
32            return bean;
33    }

34
35}

36
 

结果测试:

  1package com.ldd600.exception.test;
  2
  3import org.jmock.Expectations;
  4import org.jmock.Mockery;
  5import org.springframework.beans.factory.BeanFactory;
  6
  7import com.ldd600.exception.action.BusinessAction;
  8import com.ldd600.exception.base.BaseAppException;
  9import com.ldd600.exception.base.BaseAppRuntimeException;
 10import com.ldd600.exception.base.ConfigException;
 11import com.ldd600.exception.base.handler.ConsoleHandler;
 12import com.ldd600.exception.context.CoreContextFactory;
 13import com.ldd600.exception.dto.DefaultRequest;
 14import com.ldd600.exception.dto.DefaultResponse;
 15import com.ldd600.exception.dto.Request;
 16import com.ldd600.exception.dto.Response;
 17import com.ldd600.exception.webservice.ActionBrokerImpl;
 18
 19public class ExceptionTest extends DependencyInjectionExceptionTestCase {
 20    Mockery context = new Mockery();
 21    ActionBrokerImpl broker = new ActionBrokerImpl();
 22    final Request request = new DefaultRequest();
 23    final Response response = new DefaultResponse();
 24
 25    @Override
 26    protected String[] getConfigLocations() {
 27        return new String[] "applicationContext.xml" };
 28    }

 29
 30    public void testExceptionThrow() {
 31        final BusinessAction<Response, Request> action = context
 32                .mock(BusinessAction.class);
 33        final BeanFactory beanFactory = context.mock(BeanFactory.class);
 34        assertThrowing(new Closure() {
 35            public void run() throws Throwable {
 36                context.checking(new Expectations() {
 37                    {
 38                        allowing(beanFactory).getBean("action");
 39                        will(returnValue(action));
 40                        one(action).execute(request, response);
 41                        will(throwException(new BaseAppException()));
 42                    }

 43                }
);
 44                broker.setExceptionHandler(new ConsoleHandler());
 45                broker.setBeanFactory(beanFactory);
 46                broker.execute("action", request, response);
 47            }

 48
 49        }
, BaseAppException.class);
 50    }

 51
 52    public void testExceptionalAutoLoad() throws BaseAppException {
 53        final BeanFactory beanFactory = context.mock(BeanFactory.class);
 54        final BusinessAction<Response, Request> action = context
 55                .mock(BusinessAction.class);
 56        context.checking(new Expectations() {
 57            {
 58                allowing(beanFactory).getBean("action");
 59                will(returnValue(action));
 60                one(action).execute(request, response);
 61                will(throwException(new ConfigException()));
 62            }

 63        }
);
 64        broker.setBeanFactory(beanFactory);
 65        broker.execute("action", request, response);
 66        assertEquals(CoreContextFactory.getInstance().getExceptionContext()
 67                .getErrorCode(ConfigException.class), "LDD600-00002");
 68        context.assertIsSatisfied();
 69    }

 70
 71    public void testRuntimeException() {
 72        final BusinessAction<Response, Request> action = context
 73                .mock(BusinessAction.class);
 74        final BeanFactory beanFactory = context.mock(BeanFactory.class);
 75        assertThrowing(new Closure() {
 76            public void run() throws Throwable {
 77                context.checking(new Expectations() {
 78                    {
 79                        allowing(beanFactory).getBean("action");
 80                        will(returnValue(action));
 81                        one(action).execute(request, response);
 82                        will(throwException(new BaseAppRuntimeException()));
 83                    }

 84                }
);
 85                broker.setExceptionHandler(new ConsoleHandler());
 86                broker.setBeanFactory(beanFactory);
 87                broker.execute("action", request, response);
 88            }

 89
 90        }
, BaseAppRuntimeException.class);
 91        // test config
 92        assertEquals(CoreContextFactory.getInstance().getExceptionContext()
 93                .getErrorCode(BaseAppRuntimeException.class), "LDD600-00001");
 94        // test handler
 95        assertFalse(response.isSuccess());
 96        assertEquals(response.getErrorCode(), CoreContextFactory.getInstance()
 97                .getExceptionContext().getErrorCode(
 98                        BaseAppRuntimeException.class));
 99        context.assertIsSatisfied();
100    }

101
102    public void testCheckedException() {
103        final BusinessAction<Response, Request> action = context
104                .mock(BusinessAction.class);
105        final BeanFactory beanFactory = context.mock(BeanFactory.class);
106        assertThrowing(new Closure() {
107            public void run() throws Throwable {
108                context.checking(new Expectations() {
109                    {
110                        allowing(beanFactory).getBean("action");
111                        will(returnValue(action));
112                        one(action).execute(request, response);
113                        will(throwException(new ExceptionFaker()));
114                    }

115                }
);
116                broker.setBeanFactory(beanFactory);
117                broker.execute("action", request, response);
118            }

119
120        }
, ExceptionFaker.class);
121        // test config
122        assertEquals(CoreContextFactory.getInstance().getExceptionContext()
123                .getErrorCode(ExceptionFaker.class), "LDD600-00003");
124        // test handler
125        assertFalse(response.isSuccess());
126        assertEquals(response.getErrorCode(), CoreContextFactory.getInstance()
127                .getExceptionContext().getErrorCode(
128                        ExceptionFaker.class));
129        context.assertIsSatisfied();
130    }

131}

132


参考资料:
 

文章1http://www.onjava.com/pub/a/onjava/2006/01/11/exception-handling-framework-for-j2ee.html

文章2http://sannotations.sourceforge.net/

本文源代码:源代码下载



posted on 2008-06-30 18:27 叱咤红人 阅读(7728) 评论(8)  编辑  收藏 所属分类: Spring

评论

# re: 在Spring基础上实现自己的异常处理框架 2008-06-30 20:12 蓝剑
不错,很有参考价值!  回复  更多评论
  

# re: 在Spring基础上实现自己的异常处理框架 2008-07-01 11:48 ken.wug
我只要service的异常,可以实现吗?
一般action是表现层的一些业务逻辑,这类一场,我会直接包装后,提示用户。  回复  更多评论
  

# re: 在Spring基础上实现自己的异常处理框架 2008-07-01 14:01 ldd600
service的异常,需要继续向上抛出。
我想有两种方法吧:
1.你可以用不同种类的异常区分action和service,然后用不同的handler处理异常,一个提示给用户,另一个自己记录
2.可以在action和service之间加一个broker  回复  更多评论
  

# re: 在Spring基础上实现自己的异常处理框架 2008-07-01 14:02 ldd600
感谢您的回复! :)
@ken.wug
service的异常,需要继续向上抛出。
我想有两种方法吧:
1.你可以用不同种类的异常区分action和service,然后用不同的handler处理异常,一个提示给用户,另一个自己记录
2.可以在action和service之间加一个broker  回复  更多评论
  

# re: 在Spring基础上实现自己的异常处理框架 2008-07-03 08:59 庞永庆
★☆★诚招Java书籍作者(兼职)★☆★

Java中又分为很多方面,可以写基础入门书,或者Web Eclipse 等多个方向。

大部分作者都是异地兼职来写书,异地兼职来做是完全没有问题的。
作者可以得到如下报酬:
(1)丰厚的稿酬。
(2)书籍的署名权,和由此带来的宣传效益。
(3)能够很大程度的提高作者的人气和声望。
如果你有这方面的兴趣,可以通过下面的联系方式和我联系
books_522008@yahoo.com.cn
MSN:pyq_19852008@hotmail.com
QQ:878 298 915 注明Java
我会尽快和你联系
  回复  更多评论
  

# re: 在Spring基础上实现自己的异常处理框架 2008-07-09 10:07 pl
收藏下~  回复  更多评论
  

# re: 在Spring基础上实现自己的异常处理框架[未登录] 2008-12-23 11:25 aaa
太复杂了  回复  更多评论
  

# re: 在Spring基础上实现自己的异常处理框架 2014-04-11 17:45 最代码
最代码网站上整理了你的代码,地址:
http://www.zuidaima.com/share/1774096228535296.htm
有问题请回复  回复  更多评论
  


只有注册用户登录后才能发表评论。


网站导航: