Ответ 1
Информация об исключении составляет уже, доступная несколькими атрибутами запроса. Вы можете найти имена всех этих атрибутов в RequestDispatcher
javadoc:
-
ERROR_EXCEPTION
-javax.servlet.error.exeption
-
ERROR_EXCEPTION_TYPE
-javax.servlet.error.exception_type
-
ERROR_MESSAGE
-javax.servlet.error.message
-
ERROR_REQUEST_URI
-javax.servlet.error.request_uri
-
ERROR_SERVLET_NAME
-javax.servlet.error.servlet_name
-
ERROR_STATUS_CODE
-javax.servlet.error.status_code
Итак, вкратце, этот пример JSP должен отображать все возможные детали исключения:
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
...
<ul>
<li>Exception: <c:out value="${requestScope['javax.servlet.error.exception']}" /></li>
<li>Exception type: <c:out value="${requestScope['javax.servlet.error.exception_type']}" /></li>
<li>Exception message: <c:out value="${requestScope['javax.servlet.error.message']}" /></li>
<li>Request URI: <c:out value="${requestScope['javax.servlet.error.request_uri']}" /></li>
<li>Servlet name: <c:out value="${requestScope['javax.servlet.error.servlet_name']}" /></li>
<li>Status code: <c:out value="${requestScope['javax.servlet.error.status_code']}" /></li>
</ul>
Кроме того, вы также можете показать эту полезную информацию:
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt" %>
<jsp:useBean id="date" class="java.util.Date" />
...
<ul>
<li>Timestamp: <fmt:formatDate value="${date}" type="both" dateStyle="long" timeStyle="long" /></li>
<li>User agent: <c:out value="${header['user-agent']}" /></li>
</ul>
Конкретный экземпляр Exception
сам по себе находится в JSP, доступном только как ${exception}
, когда вы отмечаете страницу как страницу с ошибкой:
<%@ page isErrorPage="true" %>
...
${exception}
Только если вы используете EL 2.2 или новее, вы можете распечатать его стек, как показано ниже:
<%@ page isErrorPage="true" %>
...
<pre>${pageContext.out.flush()}${exception.printStackTrace(pageContext.response.writer)}</pre>
Или, если вы еще не на EL 2.2, создайте пользовательскую функцию EL:
public final class Functions {
private Functions() {}
public static String printStackTrace(Throwable exception) {
StringWriter stringWriter = new StringWriter();
exception.printStackTrace(new PrintWriter(stringWriter, true));
return stringWriter.toString();
}
}
Что зарегистрировано в /WEB-INF/functions.tld
:
<?xml version="1.0" encoding="UTF-8" ?>
<taglib
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-jsptaglibrary_2_1.xsd"
version="2.1">
<display-name>Custom Functions</display-name>
<tlib-version>1.0</tlib-version>
<uri>http://example.com/functions</uri>
<function>
<name>printStackTrace</name>
<function-class>com.example.Functions</function-class>
<function-signature>java.lang.String printStackTrace(java.lang.Throwable)</function-signature>
</function>
</taglib>
И может использоваться как
<%@ taglib prefix="my" uri="http://example.com/functions" %>
...
<pre>${my:printStackTrace(exception)}</pre>
Что касается регистрации исключения, самым простым местом будет filter, который отображается в шаблоне URL-адреса /*
и выполняет в основном следующее:
try {
chain.doFilter(request, response);
} catch (ServletException e) {
log(e.getRootCause());
throw e;
} catch (IOException e) { // If necessary? Usually not thrown by business code.
log(e);
throw e;
}