spring mvc 异常如何处理

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了spring mvc 异常如何处理相关的知识,希望对你有一定的参考价值。

一、使用@ExceptionHandler进行处理
1.创建异常基类,使用@ExceptionHandler声明异常处理
BusinessException和SystemException为自定义异常类,代码如下:

package com.twosnail.exception;

import javax.servlet.http.HttpServletRequest;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.ExceptionHandler;

@Controller
public class BasicExController
/**
* 基于@ExceptionHandler异常处理基类
* @return
*/
@ExceptionHandler
public String exception( HttpServletRequest request , Exception ex )

// 根据不同错误转向不同页面
if( ex instanceof BusinessException )
return "business-error";
else if( ex instanceof SystemException )
return "system-error";
else
return "error";




2、使所有需要异常处理的Controller都继承该类,如下所示:
public class DemoController extends BasicExController

然而,Dao层、Service层、Controller层抛出的异常(BusinessException、SystemException和其它异常)都能准确显示定义的异常处理页面,达到了统一异常处理的目标。
二、SimpleMappingExceptionResolver简单异常处理器
SimpleMappingExceptionResolver有两种配置方式,可以按自己需求而定,配置代码如下:
1、第一种,在Spring的配置文件中,增加以下内容:
在这里,可以设置跳转相应页面。
<bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
<!-- 定义默认的异常处理页面,当该异常类型的注册时使用 -->
<property name="defaultErrorView" value="error"></property>
<!-- 定义异常处理页面用来获取异常信息的变量名,默认名为exception -->
<property name="exceptionAttribute" value="ex"></property>
<!-- 定义需要特殊处理的异常,用类名或完全路径名作为key,异常也页名作为值 -->
<property name="exceptionMappings">
<props>
<prop key="com.twosnail.exception.BusinessException">business-error</prop>
<prop key="com.twosnail.exception.SystemException">system-error</prop>
</props>
</property>

<!-- 相关状态码对应的错误页面 -->
<property name="statusCodes">
<props>
<prop key="errors/500">500</prop>
<prop key="errors/404">404</prop>
</props>
</property>
<!-- 设置日志输出级别,不定义则默认不输出警告等错误日志信息 -->
<property name="warnLogCategory" value="WARN" />
<!-- 默认HTTP状态码 -->
<property name="defaultStatusCode" value="500" />
</bean>

2、第二种,通过自定义java类,继承SimpleMappingExceptionResolver
然后在Spring的配置。代码如下:
<bean id="exceptionResolver" class="com.twosnail.exception.MyselfSimpleMappingExceptionResolver">
<property name="exceptionMappings">
<props>
<prop key="com.twosnail.exception.SystemException">error/500</prop>
<prop key="com.twosnail.exception.BusinessException">error/errorpage</prop>
<prop key="java.lang.exception">error/500</prop>
</props>
</property>
</bean>

java类代码如下,在这里可以处理相应逻辑,如下,分别处理了jsp页面和json数据:
package com.twosnail.exception;

import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.handler.SimpleMappingExceptionResolver;

public class MyselfSimpleMappingExceptionResolver extends SimpleMappingExceptionResolver

@Override
protected ModelAndView doResolveException(HttpServletRequest request,
HttpServletResponse response, Object handler, Exception ex)
// Expose ModelAndView for chosen error view.
String viewName = determineViewName(ex, request);
if (viewName != null) // JSP格式返回
if (!(request.getHeader("accept").indexOf("application/json") > -1 || (request
.getHeader("X-Requested-With") != null && request
.getHeader("X-Requested-With").indexOf("XMLHttpRequest") > -1)))
// 如果不是异步请求
// Apply HTTP status code for error views, if specified.
// Only apply it if we\'re processing a top-level request.
Integer statusCode = determineStatusCode(request, viewName);
if (statusCode != null)
applyStatusCodeIfPossible(request, response, statusCode);

return getModelAndView(viewName, ex, request);
else // JSON格式返回
try
PrintWriter writer = response.getWriter();
writer.write(ex.getMessage());
writer.flush();
catch (IOException e)
e.printStackTrace();

return null;


else
return null;




总结:使用SimpleMappingExceptionResolver进行异常处理,具有集成简单、有良好的扩展性、对已有代码没有入侵性等优点,但方法1仅能获取到异常信息,若在出现异常时,对需要获取除异常以外的数据的情况不适用。
三、HandlerExceptionResolver自定义异常
1.在Spring的配置文件中,增加以下内容:
<bean id="exceptionHandler" class="com.twosnail.exception.MyExceptionHandler"/>

2.添加自定义的MyExceptionHandler类,代码如下:
在这里,单独打印出了异常路径,便于在日志中查看,在对SystemException异常进行了特殊处理:
package com.twosnail.exception;

import java.util.Map;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.HandlerExceptionResolver;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.View;
import org.springframework.web.servlet.view.RedirectView;

public class MyExceptionHandler implements HandlerExceptionResolver

public ModelAndView resolveException( HttpServletRequest request, HttpServletResponse response,
Object handler, Exception exception )

System.out.println( "【抛出异常】--异常路径为:" +
request.getServletPath() + "\\n【异常信息】--" + exception.getMessage() ) ;
//如果不是抛出的action业务异常则不处理
if( !( exception instanceof SystemException ) )
return null;


final SystemException actionE = (SystemException) exception;
ModelAndView model = null;
if( actionE.getForwardType() == SystemException.FORWARD )
//进入页面渲染
model = new ModelAndView( actionE.getModelPath(), actionE.getAttributes());
else if( actionE.getForwardType() == SystemException.REDIRECT )
model = new ModelAndView( new RedirectView( actionE.getModelPath(), true));
else
//直接返回页面内容
model = new ModelAndView( new View()
@Override
public void render(Map<String, ?> arg0, HttpServletRequest arg1,
HttpServletResponse arg2) throws Exception

arg2.setContentType( "text/html" );
arg2.setCharacterEncoding( actionE.getEncode() );
if( actionE.getResponseBody() != null )
arg2.getWriter().print( actionE.getResponseBody() );



@Override
public String getContentType()
return "text/html; charset=utf-8";

);


return model;

参考技术A 通过Spring mvc框架提供的SimpleMappingExceptionResolver来集中处理,该类可以捕获所有控制器抛出的异常,并将它映射到配置的视图中。

用法配置如下:
<bean id="exceptionResolver"
class="org.springframework.web.servlet.handler.SimpleMappingEx
ceptionResolver">
<property name="defaultErrorView">
<value>failure</value>
</property>
<property name="exceptionMappings">
<props>
<prop key="java.sql.SQLException">showDBError</prop>
<prop key="java.lang.RuntimeException">showError</prop>
</props>
</property>
</bean>本回答被提问者和网友采纳
参考技术B Catch啊

spring-boot启动过程中如何处理异常?

【中文标题】spring-boot启动过程中如何处理异常?【英文标题】:How to handle exceptions during spring-boot start-up? 【发布时间】:2019-02-18 05:13:17 【问题描述】:

这不是关于如何处理 Spring MVC 中的异常或其他任何东西。我特别需要处理 spring 开始时可能发生的异常,即在整个应用程序上下文甚至初始化之前。

对于一些背景知识,有问题的应用程序是一个允许远程访问电子设备的 IoT 节点。它内置了一个小的 h2 数据库来保存一些数据。这些数据在某些时候很不错,但对于应用程序的工作来说并不是必不可少的。

碰巧运行应用程序的设备偶尔会断电,如果在对数据库进行写操作时发生这种情况,则文件已损坏并且将出现 JdbcSQLException当应用程序尝试再次启动时抛出。 由于数据并不是真正重要的,使应用程序再次工作的最简单方法是删除数据库并让 h2 重新创建它。但为了做到这一点,我必须捕获异常,以便对它做出反应。应用程序不必必须继续启动,它将由 systemd 再次启动。我真的只需要识别异常并删除文件,就是这样。

有一种明显的方法可以做到这一点,即将 SpringApplication.run 放在 try-catch 块中。但这也很丑陋,因为我得到了我正在寻找的异常嵌套在由 h2 无法启动引起的大量 spring 异常中。

还有人建议我在实例化数据库的bean中捕获异常,但不幸的是没有bean实例化它。 DB 用作 Quartz 作业存储,因此完全由 spring 管理。它在代码中的全部存在是属性文件中的以下条目:

spring.quartz.job-store-type=jdbc
spring.quartz.properties.org.quartz.jobStore.misfireThreshold=900000
spring.datasource.name=h2
spring.datasource.url=jdbc:h2:file:$config.foldercontroller
spring.datasource.driverClassName=org.h2.Driver

我的问题是,有没有办法注册某种异常处理程序或其他方式,以便在异常发生时直接处理异常,这样我可以更轻松地识别它?

【问题讨论】:

【参考方案1】:

取决于您如何声明 bean。像这样简单地包裹 bean 有什么问题?

@Configuration
class Conf 
    @Bean
    public DB foo() throws JdbcSQLException
    
        try
        
            return new DB();
        
        catch(JdbcSQLException e)
        
            deleteDatabase();
            throw JdbcSQLException;
        
    

    public static void deleteDatabase()
    
        //...
    

【讨论】:

两个原因:一,我对 spring-boot 很烂,甚至没有考虑这一点,不幸的是,二,整个 DB 充当 Quartz Job-store 并且由 spring 本身设置。代码中没有 bean 实例化它。我不太确定自己要如何实例化它……(见原因一)。 spring是如何实例化的呢?您必须包含一些依赖项,或者您定义的一些属性,或者其他东西。请使用这些详细信息编辑您的问题。 @UncleBob 谢谢。鉴于该信息,我认为您必须使用启动器。类似于spring-boot-starter-jdbcspring-boot-starter-data-jpa。你有这样的事吗? 确实,依赖项中有 spring-boot-starter-quartz 和 spring-boot-starter-jdbc。 酷。您基本上需要这样做来覆盖自动配置:***.com/questions/28821521/…

以上是关于spring mvc 异常如何处理的主要内容,如果未能解决你的问题,请参考以下文章

在 Spring MVC 中如何处理 Ajax 请求?

在(Spring MVC)Web 请求中应该如何处理 SessionFactory 实例?

spring mvc 如何处理jquery ajax请求

Spring security 3.1如何处理不同的认证异常?

如何处理spring boot 2中@Requestbody中的对象抛出的异常

如何处理 grails spring-security-rest 插件中的自定义身份验证异常?