捕获所有Tomcat(GWT)servlet中所有未处理异常的最佳方法

时间:2012-12-27 11:45:27

标签: java gwt tomcat unhandled-exception

我将Tomcat 7和Lo4j用于所有服务器日志,使用GWT用于客户端(仅限AJAX调用)。 所有未处理的异常都记录在我的catalina.log中。

现在我想捕获所有异常并添加一些用户特定的Tomcat SessionData。

有几种方法:

  • 尝试捕获所有servlet(必须有更好的解决方案)。
  • http://tomcat.apache.org/tomcat-7.0-doc/aio.html:我必须更改我的连接器,我不知道是否可以在事件处理程序中使用Tomcat会话(EventType.ERROR)。
  • 更好的方式?

实现这一目标的最佳方式是什么?

3 个答案:

答案 0 :(得分:7)

根据我对您的问题的理解,您可以尝试使用以下两种方法中的至少一种:

基本日志Servlet

如果您可以访问所有servlet的源代码,则可以使用一个基本的超级servlet进行一些重构,该servlet负责日志记录/使用AJAX透明地工作的任何请求,没有错误转发指令,并且没有全局异常处理程序。假设您使用service(ServletRequest,ServletResponse)作为servlet入口点(但是您可以为每个do*()方法执行以下操作),然后您可以创建一个抽象的超级servlet并从中继承您的servlet。

<servlet>
    <servlet-name>servlet1</servlet-name>
    <servlet-class>stackoverflow.Servlet1</servlet-class>
</servlet>

<servlet>
    <servlet-name>servlet2</servlet-name>
    <servlet-class>stackoverflow.Servlet2</servlet-class>
</servlet>

<servlet-mapping>
    <servlet-name>servlet1</servlet-name>
    <url-pattern>servlet1</url-pattern>
</servlet-mapping>

<servlet-mapping>
    <servlet-name>servlet2</servlet-name>
    <url-pattern>servlet2</url-pattern>
</servlet-mapping>
public abstract class BasicServlet extends HttpServlet {

    /**
     * Won't let it be {@code abstract} - we don't want to force any sub-servlet to implement this method.
     */
    protected void doService(ServletRequest request, ServletResponse response) {
    }

    @Override
    public final void service(ServletRequest request, ServletResponse response) {
        try {
            doService(request, response);
        } catch ( Throwable ex ) {
            err.println(ex.getMessage());
        }
    }

}
public final class Servlet1 extends BasicServlet {

    @Override
    protected void doService(ServletRequest request, ServletResponse response) {
        out.println("I'm servlet #1");
    }

}
public final class Servlet2 extends BasicServlet {

    @Override
    protected void doService(ServletRequest request, ServletResponse response) {
        out.println("I'm servlet #2");
    }

}

此方法的一个优点是除了更改servlet类而不依赖于外部配置或上下文之外,您不需要配置任何其他内容。缺点是你总是必须扩展BasicServlet

过滤

我目前没有对它进行实际测试,有关详细信息,请参阅http://docs.oracle.com/javaee/6/api/javax/servlet/Filter.html。过滤器允许拦截每个请求(我们在调试和将异常写入公共日志文件时使用JSP的这种过滤器实现)。缺点是不能保证过滤器可以覆盖每个异常/情况,例如,如果任何过滤器在您自己的过滤器之前。

<filter>
    <filter-name>exceptionLoggingFilter</filter-name>
    <filter-class>stackoverflow.ExceptionLoggingFilter</filter-class>
</filter>

<filter-mapping>
    <filter-name>exceptionLoggingFilter</filter-name>
    <url-pattern>*</url-pattern> <!-- we will process every request -->
</filter-mapping>
public final class ExceptionLoggingFilter implements Filter {

    @Override
    public void init(FilterConfig filterConfig) {
    }

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) {
        try {
            filterChain.doFilter(request, response);
        } catch ( Throwable ex ) {
            err.println(ex);
        }
    }

    @Override
    public void destroy() {
    }

}

希望这有帮助。

答案 1 :(得分:3)

刚刚覆盖GWT功能doUnexpectedFailure有效。

@Override
protected void doUnexpectedFailure(Throwable t) {
  ServerLog.error(t.getMessage(), t);
  super.doUnexpectedFailure(t);
}

答案 2 :(得分:2)

1)您可以为您的webapp定义错误页面,如下所示:

<error-page>
        <exception-type>java.lang.Throwable</exception-type>
        <location>/error</location>
</error-page>

然后你可以在/error绑定另一个servlet并在那里处理异常。

2)您可以为每个HTTP连接器线程setUncaughtExceptionHandler。您可以将此技术与servlet过滤器一起使用,该过滤器将包含对当前HttpRequest的引用(例如,通过本地线程)。顺便说一下,这不适用于异步I / O.