基于spring注解AOP的異常處理

一、前言

  項目剛剛開發的時候,并沒有做好充足的準備。開發到一定程度的時候才會想到還有一些問題沒有解決。就比如今天我要說的一個問題:異常的處理。寫程序的時候一般都會通過try...catch...finally對異常進行處理,但是我們真的能在寫程序的時候處理掉所有可能發生的異常嗎? 以及發生異常的時候執行什么邏輯,返回什么提示信息,跳轉到什么頁面,這些都是要考慮到的。

二、基于@ControllerAdvice(加強的控制器)的異常處理

  參考文檔:http://jinnianshilongnian.iteye.com/blog/1866350

  @ControllerAdvice注解內部使用@ExceptionHandler、@InitBinder、@ModelAttribute注解的方法應用到所有的?@RequestMapping注解的方法。本例子中使用ExceptionHandler應用到所有@RequestMapping注解的方法,處理發生的異常。

  示例代碼:

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.util.StringUtils;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;import com.hjz.exception.ServiceException;
import com.hjz.exception.utils.ExceptionUtils;@ResponseBody
public class ExceptionAdvice {private static final Logger LOGGER = LoggerFactory.getLogger(ExceptionAdvice.class);/*** 攔截web層異常,記錄異常日志,并返回友好信息到前端* 目前只攔截Exception,是否要攔截Error需再做考慮** @param e 異常對象* @return 異常提示*/@ExceptionHandler(Exception.class)public ResponseEntity<String> handleException(Exception e) {//不需要再記錄ServiceException,因為在service異常切面中已經記錄過if (!(e instanceof ServiceException)) {LOGGER.error(ExceptionUtils.getExcTrace(e));}HttpHeaders headers = new HttpHeaders();headers.set("Content-type", "text/plain;charset=UTF-8");headers.add("icop-content-type", "exception");String message = StringUtils.isEmpty(e.getMessage()) ? "系統異常!!" : e.getMessage();return new ResponseEntity<>(message, headers, HttpStatus.OK);}
}

  如果不起作用,請檢查 spring-mvc的配置文件,是否有ControllerAdvice的如下配置

<context:component-scan base-package="com.xxx.xx" use-default-filters="false">  <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>  <context:include-filter type="annotation" expression="org.springframework.web.bind.annotation.ControllerAdvice"/>  </context:component-scan>  

?  附上:Spring MVC的Controller統一異常處理:HandlerExceptionResolver

三、基于AOP的異常處理

  1.處理controller層的異常?WebExceptionAspect.java

import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;import com.hjz.exception.ServiceException;
import com.hjz.exception.utils.ExceptionUtils;import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;/*** web異常切面* 默認spring aop不會攔截controller層,使用該類需要在spring公共配置文件中注入改bean,* 另外需要配置<aop:aspectj-autoproxy proxy-target-class="true"/>*/
@Aspect
public class WebExceptionAspect {private static final Logger LOGGER = LoggerFactory.getLogger(WebExceptionAspect.class);@Pointcut("@annotation(org.springframework.web.bind.annotation.RequestMapping)")private void webPointcut() {}/*** 攔截web層異常,記錄異常日志,并返回友好信息到前端* 目前只攔截Exception,是否要攔截Error需再做考慮** @param e 異常對象*/@AfterThrowing(pointcut = "webPointcut()", throwing = "e")public void handleThrowing(Exception e) {//不需要再記錄ServiceException,因為在service異常切面中已經記錄過if (!(e instanceof ServiceException)) {LOGGER.error(ExceptionUtils.getExcTrace(e));}String errorMsg = StringUtils.isEmpty(e.getMessage()) ? "系統異常" : e.getMessage();writeContent(errorMsg);}/*** 將內容輸出到瀏覽器** @param content 輸出內容*/private void writeContent(String content) {HttpServletResponse response = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getResponse();response.reset();response.setCharacterEncoding("UTF-8");response.setHeader("Content-Type", "text/plain;charset=UTF-8");response.setHeader("icop-content-type", "exception");PrintWriter writer = null;try {writer = response.getWriter();} catch (IOException e) {e.printStackTrace();}writer.print(content);writer.flush();writer.close();}
}

  2.處理service層的異常ServiceExceptionAspect?.java

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;import com.hjz.exception.ServiceException;
import com.hjz.exception.utils.ExceptionUtils;@Aspect
public class ServiceExceptionAspect {private static final Logger LOGGER = LoggerFactory.getLogger(ServiceExceptionAspect.class);/*** @within(org.springframework.stereotype.Service),攔截帶有 @Service 注解的類的所有方法* @annotation(org.springframework.web.bind.annotation.RequestMapping),攔截帶有@RquestMapping的注解方法*/@Pointcut("@within(org.springframework.stereotype.Service) && execution(public * *(..))")private void servicePointcut() {}/*** 攔截service層異常,記錄異常日志,并設置對應的異常信息* 目前只攔截Exception,是否要攔截Error需再做考慮** @param e 異常對象*/@AfterThrowing(pointcut = "servicePointcut()", throwing = "e")public void handle(JoinPoint point, Exception e) {LOGGER.error(ExceptionUtils.getExcTrace(e));String signature = point.getSignature().toString();String errorMsg = getMessage(signature) == null ? (StringUtils.isEmpty(e.getMessage()) ? "服務異常" : e.getMessage()) : getMessage(signature);throw new ServiceException(errorMsg, e);}/*** 獲取方法簽名對應的提示消息** @param signature 方法簽名* @return 提示消息*/private String getMessage(String signature) {return null;}
}

?  3.使用方式,在spring的公共配置文件中加入如下配置:

<aop:aspectj-autoproxy proxy-target-class="true" />
<bean class="com.hjz.exception.aspect.ServiceExceptionAspect" />
<bean class="com.hjz.exception.aspect.WebExceptionAspect" />

  或者 自定義一個 注冊類,ServiceExceptionAspect.java和WebExceptionAspect.java都加入@Component注解

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;/*** 異常相關bean注冊類*/
@Configuration
@EnableAspectJAutoProxy
@ComponentScan("com.hjz.exception.aspect")
public class ExceptionConfig {}

  注:spring 公共配置文件中的配置 改成?<bean class="com.hjz.exception.config.ExceptionConfig"/>,如果controller層的異常無法攔截,請將配置換到springmvc的配置文件中,原因請見(SpringMVC關于AOP攔截controller的注意事項

@Aspect
@Component
public class WebExceptionAspect {..........  
}@Aspect
@Component
public class ServiceExceptionAspect {.........
}

四、疑惑

?  @within(org.springframework.stereotype.Service),攔截帶有 @Service 注解的類的所有方法
?  @annotation(org.springframework.web.bind.annotation.RequestMapping),攔截帶有@RquestMapping的注解方法

五、測試

  分別編寫controller層和service層的異常測試類。這個很簡單,在方法里簡單的拋一下異常就可以了。最后驗證一下,異常發生的時候有沒有 執行?@AfterThrowing對應的方法就好了。具體還是看我寫的demo吧,嘿嘿嘿!!!

  完整項目下載地址:https://github.com/hjzgg/Spring-annotation-AOP-Exception_handling

?

轉載于:https://www.cnblogs.com/hujunzheng/p/6255463.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/531279.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/531279.shtml
英文地址,請注明出處:http://en.pswp.cn/news/531279.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Kettle之數據抽取、轉換、裝載

Kettle 官網 ETL利器Kettle實戰應用解析系列 利用kettle組件導入excel文件到數據庫 kettle中實現動態SQL查詢 java中調用kettle轉換文件 kettle 7.x版本下載&#xff1a;https://pan.baidu.com/s/1nvnzzCH  密碼&#xff1a;6f5c mac 下運行spoon.sh,  windows下為spoon.bat…

webpack+react+redux+es6開發模式---續

一、前言 之前介紹了webpackreactreduxes6開發模式 &#xff0c;這個項目對于一個獨立的功能節點來說是沒有問題的。假如伴隨著源源不斷的需求&#xff0c;前段項目會涌現出更多的功能節點&#xff0c;需要獨立部署運行。為了更好地管理這些獨立的功能節點&#xff0c;我們需要…

JdbcTemplate使用小結

org.springframework.jdbc.core.JdbcTemplate.query(String sql, Object[] args, RowMapper<StaffUnionVO> rowMapper) throws DataAccessException 1.自定義rowMapper public class StaffUnionVO implements RowMapper<StaffUnionVO>, Serializable {private stat…

RabbitMQ安裝和使用(和Spring集成)

一、安裝Rabbit MQ   Rabbit MQ 是建立在強大的Erlang OTP平臺上&#xff0c;因此安裝Rabbit MQ的前提是安裝Erlang。通過下面兩個連接下載安裝3.2.3 版本&#xff1a; 下載并安裝 Eralng OTP For Windows (vR16B03)運行安裝 Rabbit MQ Server Windows Installer (v3.2.3)具體…

單點登錄實現(spring session+redis完成session共享)

一、前言 項目中用到的SSO&#xff0c;使用開源框架cas做的。簡單的了解了一下cas&#xff0c;并學習了一下 單點登錄的原理&#xff0c;有興趣的同學也可以學習一下&#xff0c;寫個demo玩一玩。 二、工程結構 我模擬了 sso的客戶端和sso的服務端&#xff0c; sso-core中主要是…

maven deploy上傳私服出錯

error 內容如下 Failed to execute goal org.apache.maven.plugins:maven-deploy-plugin:2.5: deploy (default-deploy) on project XXX pom文件增加如下配置 <build><plugins><plugin><groupId>org.apache.maven.plugins</groupId><artifa…

加密策略

一、前言 這兩天研究了一下項目中的密碼加密&#xff0c;可以說得上是學到了很多。下面來大致說一下。 二、常用加密 1.單向加密算法 單向加密算法主要用來驗證數據傳輸的過程中&#xff0c;是否被篡改過。 BASE64 嚴格地說&#xff0c;屬于編碼格式&#xff0c;而非加密算法 …

Spring Data JPA: 實現自定義Repository

一、前言 由于項目中的 實體&#xff08;entity&#xff09;默認都是繼承一個父類&#xff08;包含一些公共的屬性&#xff0c;比如創建時間&#xff0c;修改時間&#xff0c;是否刪除&#xff0c;主鍵id&#xff09;。為了實現邏輯刪除&#xff0c;一般會自己實現RepositoryFa…

java json和對象互相裝換

java json和對象互相裝換 1.com.alibaba.fastjson.JSON 2.com.fasterxml.jackson.databind.ObjectMapper 轉載于:https://www.cnblogs.com/hujunzheng/p/6593491.html

js沖刺一下

js中__proto__和prototype的區別和關系 1.對象有屬性__proto__,指向該對象的構造函數的原型對象。  2.方法除了有屬性__proto__,還有屬性prototype&#xff0c;prototype指向該方法的原型對象。 深入淺出妙用 Javascript 中 apply、call、bind ***兩道面試題*** 關于js中偽數…

springmvc 源碼分析

DispatcherServlet的初始化流程 HandlerMapping - RequestMappingHandlerMapping初始化 DefaultAnnotationHandlerMapping 和RequestMappingHandlerMapping RequestToViewNameTranslator請求到視圖名稱的轉換 (如果沒有responsebody&#xff0c;并且沒有返回一個view&#xff0…

EntityManager的使用

1、最基礎的查詢 CriteriaBuilder cb entityManager.getCriteriaBuilder(); CriteriaQuery<User> cq cb.createQuery(User.class); Root<User> root cq.from(User.class); //from User cq.select(root); //select * from User javax.persistence.criteria.Predi…

Jackson序列化實例

參考文章 Jackson使用ContextualSerializer在序列化時獲取字段注解的屬性 使用BeanSerializerModifier定制jackson的自定義序列化(null值的處理) 關于使用ContextualSerializer的補充 BeanSerializerFactory中有如下代碼&#xff0c; 關于設置SerializerModifier&#xff0c;如…

maven自定義腳手架(快速生成項目)

Maven之自定義archetype生成項目骨架 利用腳手架生成 新項目 命令行方式 mvn archetype:generate \ -DarchetypeGroupIdcom.xxx \ -DarchetypeArtifactIdarchetype-spring-boot \ -DarchetypeVersion1.0.0 \ -DgroupIdcom.xxx \ -DartifactIddemo-archetype-generate \ -Dversi…

JSONObject 自定義過濾配置

一、自定義過濾器說明 PropertyPreFilter 根據PropertyName判斷是否序列化   PropertyFilter 根據PropertyName和PropertyValue來判斷是否序列化   NameFilter 修改Key&#xff0c;如果需要修改Key,process返回值則可   ValueFilter 修改Value   BeforeFilter 序列化時…

springmvc防止重復提交攔截器

一、攔截器實現&#xff0c;ResubmitInterceptorHandler.java import org.apache.commons.lang3.StringUtils; import org.springframework.stereotype.Component; import org.springframework.web.method.HandlerMethod; import org.springframework.web.servlet.handler.Hand…

Mac idea 快捷鍵

Mac鍵盤符號和修飾鍵說明 ? Command? Shift? Option? Control?? Return/Enter? Delete? 向前刪除鍵&#xff08;FnDelete&#xff09;↑ 上箭頭↓ 下箭頭← 左箭頭→ 右箭頭? Page Up&#xff08;Fn↑&#xff09;? Page Down&#xff08;Fn↓&#xff09;Home Fn ←…

cas4.2.7實現單點登錄

準備前參考&#xff1a;  cas server下載地址 cas client 下載地址 安全cookie setSecure詳解 Spring通過構造方法注入的四種方式 cas 學習博文 自定義登錄頁和登錄認證 cas server端的login-webflow詳細流程 CAS服務端自定義數據庫認證用戶 準備工作 1. cas server下載之后解…

log4j之log4j2.xml使用

依賴jar包 log4j-api-2.6.2.jar log4j-core-2.6.2.jar log4j-slf4j-impl-2.6.2.jar slf4j-api-1.7.12.jar 在resources目錄下新建log4j2.xml&#xff0c;內容如下。 <?xml version"1.0" encoding"UTF-8"?><!--status : 這個用于設置log4j2自身內…

cas4.2.7與shiro進行整合

準備工作 cas單點登錄開始前準備&#xff0c;請參考cas4.2.7實現單點登錄。 與shiro進行整合 注&#xff1a;準備工作的基礎上&#xff0c;對cas客戶端進行如下改進。 引入相關jar包 shiro-cas-1.2.6.jar shiro-core-1.2.6.jar shiro-spring-1.2.6.jar shiro-web-1.2.6.jar web…