Java Web 错误/异常处理页面
Posted sp42a
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java Web 错误/异常处理页面相关的知识,希望对你有一定的参考价值。
发生服务器 500 异常,如果默认方式处理,则是将异常捕获之后调到 Tomcat 缺省的异常页面,如下图所示。
不论哪个网站都是一样的,所以为了满足自定义的需要,Tomcat 也允许自定义样式的。也就是在 web.xml 文件中配置:
<error-page> <error-code>500</error-code> <location>/error.jsp</location> </error-page>
首先说说自带的逻辑。如果某个 JSP 页面在执行的过程中出现了错误, 那么 JSP 引擎会自动产生一个异常对象,如果这个 JSP 页面指定了另一个 JSP 页面为错误处理程序,那么 JSP 引擎会将这个异常对象放入到 request 对象中,传到错误处理程序中。如果大家有写 Servlet 的印象,这是和那个 javax.servlet.forward.request_uri 一个思路,保留了原来请求的路径而不是 JSP 页面的那个路径。在错误处理程序里,因为 page 编译指令的 isErrorPage 属性的值被设为 true,那么 JSP 引擎会自动声明一个 exception 对象,这个 exception 对象从 request 对象所包含的 HTTP 参数中获得。
request 对象中包含的异常信息非常丰富,如下所示:
javax.servlet.error.status_code 类型为Integer 错误状态代码 javax.servlet.error.exception_type 类型为Class 异常的类型 javax.servlet.error.message 类型为String 异常的信息 javax.servlet.error.exception 类型为Throwable 异常类 javax.servlet.error.request_uri 类型为String 异常出现的页面 javax.servlet.error.servlet_name 类型为String 异常出现的servlet名你可以用 Java 语句 request.getAttribute("javax.servlet.error.status_code") 获取,也可以在 JSP 页面中通过 EL 表达式获取,如 ${requestScope["javax.servlet.error.status_code
"]}。
这个自定义错误页面虽然简单,JSP 本身也有很好的封装结果,我也看过别人不少的资源,但细究之下也有不少“学问”,于是我想重新再”磨磨这个轮子“——首先 location 是一个 jsp 页面,也可以是 servlet,不过万一 servlet 也有可能启动不起来的话那就使用简单的 jsp 页面就好了。我们通过 jsp 页面定义内部类的方法,达到页面与逻辑的分离(无须编写 servlet)。其余的思路如下:
- 在 JSP 里面完成 ErrorHandler 类,另有页面调用这个 ErrorHandler 类
- 不但可以接受 JSP 页面的错误,也可接受 servlet 的控制器传递的错误,并且提取尽量多信息
- 全部内容先写到内存,然后分别从两个输出流再输出到页面和文件
- 把错误信息输出到网页的同时,简单加几句话,可以把网页上的信息也写一份到数据库或者文本
- 可以返回 html/JSON/XML
实现代码如下:
/** * 异常处理类 */ class ErrorHandler { // 全部内容先写到内存,然后分别从两个输出流再输出到页面和文件 private ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(); private PrintStream printStream = new PrintStream(byteArrayOutputStream); /** * 收集错误信息 * @param request * @param exception * @param out */ public ErrorHandler(HttpServletRequest request, Throwable exception, JspWriter out) { setRequest(request); setException(exception); if(out != null) { try { out.print(byteArrayOutputStream); // 输出到网页 } catch (IOException e) { e.printStackTrace(); } } log(request); if(byteArrayOutputStream != null) try { byteArrayOutputStream.close(); } catch (IOException e) { e.printStackTrace(); } if(printStream != null) printStream.close(); } /** * * @param request */ private void setRequest(HttpServletRequest request) { printStream.println(); printStream.println("用户账号:" + request.getSession().getAttribute("userName")); printStream.println("访问的路径: " + getInfo(request, "javax.servlet.forward.request_uri", String.class)); printStream.println("出错页面地址: " + getInfo(request, "javax.servlet.error.request_uri", String.class)); printStream.println("错误代码: " + getInfo(request, "javax.servlet.error.status_code", int.class)); printStream.println("异常的类型: " + getInfo(request, "javax.servlet.error.exception_type", Class.class)); printStream.println("异常的信息: " + getInfo(request, "javax.servlet.error.message", String.class)); printStream.println("异常servlet: " + getInfo(request, "javax.servlet.error.servlet_name", String.class)); printStream.println(); // 另外两个对象 getInfo(request, "javax.servlet.jspException", Throwable.class); getInfo(request, "javax.servlet.forward.jspException", Throwable.class); Map<String, String[]> map = request.getParameterMap(); for (String key : map.keySet()) { printStream.println("请求中的 Parameter 包括:"); printStream.println(key + "=" + request.getParameter(key)); printStream.println(); } for (Cookie cookie : request.getCookies()){ // cookie.getValue() printStream.println("请求中的 Cookie 包括:"); printStream.println(cookie.getName() + "=" + cookie.getValue()); printStream.println(); } } /** * * @param exception */ private void setException(Throwable exception) { if (exception != null) { printStream.println("异常信息"); printStream.println(exception.getClass() + " : " + exception.getMessage()); printStream.println(); printStream.println("堆栈信息"); exception.printStackTrace(printStream); printStream.println(); } } /** * * @param request */ private void log(HttpServletRequest request) { File dir = new File(request.getSession().getServletContext().getRealPath("/errorLog")); if (!dir.exists()) { dir.mkdir(); } String timeStamp = new java.text.SimpleDateFormat("yyyyMMddhhmmssS").format(new Date()); File file = new File(dir.getAbsolutePath() + File.separatorChar + "error-" + timeStamp + ".txt"); // try(FileOutputStream fileOutputStream = new FileOutputStream(file); // PrintStream ps = new PrintStream(fileOutputStream)){// 写到文件 // ps.print(byteArrayOutputStream); // } catch (FileNotFoundException e) { // e.printStackTrace(); // } catch (IOException e) { // e.printStackTrace(); // } catch (Exception e){ // e.printStackTrace(); // } } /** * * @param request * @param key * @param type * @return */ @SuppressWarnings("unchecked") private <T> T getInfo(HttpServletRequest request, String key, Class<T> type){ Object obj = request.getAttribute(key); return obj == null ? null : (T) obj; } }
这样就可以完成异常的控制了。
以上是关于Java Web 错误/异常处理页面的主要内容,如果未能解决你的问题,请参考以下文章