色综合图-色综合图片-色综合图片二区150p-色综合图区-玖玖国产精品视频-玖玖香蕉视频

您的位置:首頁技術文章
文章詳情頁

在spring中手寫全局異常攔截器

瀏覽:2日期:2023-08-02 13:16:22

為什么要重復造輪子

你可能會問,Spring已經自帶了全局異常攔截,為什么還要重復造輪子呢?

這是個好問題,我覺得有以下幾個原因

裝逼 Spring的全局異常攔截只是針對于Spring MVC的接口,對于你的RPC接口就無能為力了 無法定制化 除了寫業務代碼,我們其實還能干點別的事

我覺得上述理由已經比較充分的解答了為什么要重復造輪子,接下來就來看一下怎么造輪子

造個什么樣的輪子?

我覺得全局異常攔截應該有如下特性

使用方便,最好和spring原生的使用方式一致,降低學習成本 能夠支持所有接口 調用異常處理器可預期,比如說定義了RuntimeException的處理器和Exception的處理器,如果這個時候拋出NullPointException,這時候要能沒有歧義的選擇預期的處理器

如何造輪子?

由于現在的應用基本上都是基于spring的,因此我也是基于SpringAop來實現全局異常攔截

首先先定義幾個注解

@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented@Componentpublic @interface ExceptionAdvice {}@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface ExceptionHandler { Class<? extends Throwable>[] value();}@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface ExceptionIntercept {}

@ExceptionAdvice 的作用是標志定義異常處理器的類,方便找到異常處理器

@ExceptionHandler 的作用是標記某個方法是處理異常的,里面的值是能夠處理的異常類型

@ExceptionIntercept 的作用是標記需要異常攔截的方法

接下來定義統一返回格式,以便出現錯誤的時候統一返回

@Datapublic class BaseResponse<T> { private Integer code; private String message; private T data; public BaseResponse(Integer code, String message) { this.code = code; this.message = message; }}

然后定義一個收集異常處理器的類

public class ExceptionMethodPool { private List<ExceptionMethod> methods; private Object excutor; public ExceptionMethodPool(Object excutor) { this.methods = new ArrayList<ExceptionMethod>(); this.excutor = excutor; } public Object getExcutor() { return excutor; } public void add(Class<? extends Throwable> clazz, Method method) { methods.add(new ExceptionMethod(clazz, method)); } //按序查找能夠處理該異常的處理器 public Method obtainMethod(Throwable throwable) { return methods.stream().filter(e -> e.getClazz().isAssignableFrom(throwable.getClass())).findFirst().orElseThrow(() ->new RuntimeException('沒有找到對應的異常處理器')).getMethod(); } @AllArgsConstructor @Getter class ExceptionMethod { private Class<? extends Throwable> clazz; private Method method; }}

ExceptionMethod 里面有兩個屬性

clazz:這個代表著能夠處理的異常 method:代表著處理異常調用的方法

ExceptionMethodPool 里面按序存放所有異常處理器,excutor是執行這些異常處理器的對象

接下來把所有定義的異常處理器收集起來

@Componentpublic class ExceptionBeanPostProcessor implements BeanPostProcessor { private ExceptionMethodPool exceptionMethodPool; @Autowired private ConfigurableApplicationContext context; @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { Class<?> clazz = bean.getClass(); ExceptionAdvice advice = clazz.getAnnotation(ExceptionAdvice.class); if (advice == null) return bean; if (exceptionMethodPool != null) throw new RuntimeException('不允許有兩個異常定義類'); exceptionMethodPool = new ExceptionMethodPool(bean); //保持處理異常方法順序 Arrays.stream(clazz.getDeclaredMethods()).filter(method -> method.getAnnotation(ExceptionHandler.class) != null).forEach(method -> { ExceptionHandler exceptionHandler = method.getAnnotation(ExceptionHandler.class); Arrays.stream(exceptionHandler.value()).forEach(c -> exceptionMethodPool.add(c,method));}); //注冊進spring容器 context.getBeanFactory().registerSingleton('exceptionMethodPool',exceptionMethodPool); return bean; }}

ExceptionBeanPostProcessor 通過實現BeanPostProcessor 接口,在bean初始化之前,把所有異常處理器塞進 ExceptionMethodPool,并把其注冊進Spring容器

然后定義異常處理器

@Componentpublic class ExceptionProcessor { @Autowired private ExceptionMethodPool exceptionMethodPool; public BaseResponse process(Throwable e) { return (BaseResponse) FunctionUtil.computeOrGetDefault(() ->{ Method method = exceptionMethodPool.obtainMethod(e); method.setAccessible(true); return method.invoke(exceptionMethodPool.getExcutor(),e); },new BaseResponse(0,'未知錯誤')); }}

這里應用了我自己通過函數式編程封裝的一些語法糖,有興趣的可以看下

最后通過AOP進行攔截

@Aspect@Componentpublic class ExceptionInterceptAop { @Autowired private ExceptionProcessor exceptionProcessor; @Pointcut('@annotation(com.example.exception.intercept.ExceptionIntercept)') public void pointcut() { } @Around('pointcut()') public Object around(ProceedingJoinPoint point) { return computeAndDealException(() -> point.proceed(),e -> exceptionProcessor.process(e)); } public static <R> R computeAndDealException(ThrowExceptionSupplier<R> supplier, Function<Throwable, R> dealFunc) { try { return supplier.get(); } catch (Throwable e) { return dealFunc.apply(e); } } @FunctionalInterface public interface ThrowExceptionSupplier<T> { T get() throws Throwable; }}

到這里代碼部分就已經完成了,我們來看下如何使用

@ExceptionAdvicepublic class ExceptionConfig { @ExceptionHandler(value = NullPointerException.class) public BaseResponse process(NullPointerException e){ return new BaseResponse(0,'NPE'); } @ExceptionHandler(value = Exception.class) public BaseResponse process(Exception e){ return new BaseResponse(0,'Ex'); } }@RestControllerpublic class TestControler { @RequestMapping('/test') @ExceptionIntercept public BaseResponse test(@RequestParam('a') Integer a){ if (a == 1){ return new BaseResponse(1,a+''); } else if (a == 2){ throw new NullPointerException(); } else throw new RuntimeException(); }}

我們通過@ExceptionAdvice標志定義異常處理器的類,然后通過@ExceptionHandler標注處理異常的方法,方便收集

最后在需要異常攔截的方法上面通過@ExceptionIntercept進行異常攔截

我沒有使用Spring那種匹配最近父類的方式尋找匹配的異常處理器,我覺得這種設計是一個敗筆,理由如下

代碼復雜 不能一眼看出要去調用哪個異常處理器,尤其是定義的異常處理器非常多的時候,要是弄多個定義類就更不好找了,可能要把所有的處理器看完才知道應該調用哪個

出于以上考慮,我只保留了一個異常處理器定義類,并且匹配順序和方法定義順序一致,從上到下依次匹配,這樣只要找到一個能夠處理的處理器,那么就知道了會如何調用

原創不易,如果覺得對你有幫助,麻煩點個贊!

我會不定期分享一些有意思的技術,點個關注不迷路-。 -

以上就是在spring中手寫全局異常攔截器的詳細內容,更多關于spring 全局異常攔截的資料請關注好吧啦網其它相關文章!

標簽: Spring
相關文章:
主站蜘蛛池模板: 亚洲人成网站观看在线播放 | 亚洲在线看| 久久精品成人 | 写真片福利视频在线播放 | 亚洲综合色一区二区三区另类 | 亚洲涩涩精品专区 | 精品国产欧美一区二区最新 | 正在播放国产大学生情侣 | 国产精品综合久成人 | 亚洲精品久久久久中文字幕一区 | 欧美在线视频免费观看 | 韩国精品一区二区三区四区五区 | 欧美亚洲国产片在线观看 | 国产精品亚洲专一区二区三区 | 美毛片| 国产成人精品亚洲777图片 | 免费特黄一级欧美大片在线看 | 性生大片一级毛片免费观看 | 欧美老妇免费做爰视频 | 中文字幕福利 | 久99频这里只精品23热 视频 | 91福利国产在线观一区二区 | 久久久视频在线 | 免费黄色三级网站 | 久久亚洲综合中文字幕 | 九九免费精品视频 | 福利视频在线午夜老司机 | 国产午夜精品久久久久小说 | 三级黄色片在线免费观看 | 秀人网私拍福利视频在线 | 色爽爽爽爽爽爽爽爽 | 午夜在线视频一区二区三区 | 97久久草草超级碰碰碰 | 欧美孕妇孕交 | 久久综久久美利坚合众国 | 亚洲99久久久久综合 | 欧美成人手机在线 | 国产精品久久久久久久久久98 | 日韩精品国产一区 | 亚洲欧美日韩国产专区一区 | 亚洲bt欧美bt国产bt |