一、前言

项目刚刚开发的时候,并没有做好充足的准备。开发到一定程度的时候才会想到还有一些问题没有解决。就比如今天我要说的一个问题:异常的处理。写程序的时候一般都会通过try...catch...finally对异常进行处理,但是我们真的能在写程序的时候处理掉所有可能发生的异常吗? 以及发生异常的时候执行什么逻辑,返回什么提示信息,跳转到什么页面,这些都是要考虑到的。
二、基于@ControllerAdvice(加强的控制器)的异常处理
@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.sishuok.es" 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>
三、基于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 {
}
@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吧,嘿嘿嘿!!!
完整项目下载地址:Spring-AOP_jb51.rar
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# spring
# aop异常处理
# spring4
# aop处理异常
# java Aop实现自动填充字段值示例
# java开发AOP基础JdkDynamicAopProxy
# java开发AOP面向切面编程入门
# Java Spring AOP之PointCut案例详解
# Java aop面向切面编程(aspectJweaver)案例详解
# Java JDK动态代理(AOP)用法及实现原理详解
# java捕获AOP级别的异常并将其传递到Controller层
# 再做
# 配置文件
# 不需
# 要再
# 下载地址
# 才会
# 能在
# 提示信息
# 看我
# 很简单
# 考虑到
# 自定义
# 一个问题
# 这些都是
# 还有一些
# 可能发生
# 就比
# 大家多多
# 就可以
# 跳转到
相关文章:
如何在Windows服务器上快速搭建网站?
如何通过山东自助建站平台快速注册域名?
建站之星代理如何优化在线客服效率?
,巨量百应是干嘛的?
如何破解联通资金短缺导致的基站建设难题?
广州美橙建站如何快速搭建多端合一网站?
如何获取上海专业网站定制建站电话?
广州顶尖建站服务:企业官网建设与SEO优化一体化方案
视频网站app制作软件,有什么好的视频聊天网站或者软件?
建站VPS选购需注意哪些关键参数?
深圳网站制作平台,深圳市做网站好的公司有哪些?
如何通过VPS建站无需域名直接访问?
建站主机与服务器功能差异如何区分?
黑客入侵网站服务器的常见手法有哪些?
昆明网站制作哪家好,昆明公租房申请网上登录入口?
魔毅自助建站系统:模板定制与SEO优化一键生成指南
建站主机无法访问?如何排查域名与服务器问题
交易网站制作流程,我想开通一个网站,注册一个交易网址,需要那些手续?
,怎么在广州志愿者网站注册?
如何实现建站之星域名转发设置?
早安海报制作网站推荐大全,企业早安海报怎么每天更换?
如何通过老薛主机一键快速建站?
如何在IIS中新建站点并解决端口绑定冲突?
太平洋网站制作公司,网络用语太平洋是什么意思?
东莞专业制作网站的公司,东莞大学生网的网址是什么?
如何快速查询域名建站关键信息?
常州企业网站制作公司,全国继续教育网怎么登录?
网站制作大概多少钱一个,做一个平台网站大概多少钱?
如何规划企业建站流程的关键步骤?
独立制作一个网站多少钱,建立网站需要花多少钱?
建站之星好吗?新手能否轻松上手建站?
如何快速搭建响应式可视化网站?
大型企业网站制作流程,做网站需要注册公司吗?
建站主机服务器选购指南:轻量应用与VPS配置解析
ppt制作免费网站有哪些,ppt模板免费下载网站?
如何有效防御Web建站篡改攻击?
如何将凡科建站内容保存为本地文件?
西安市网站制作公司,哪个相亲网站比较好?西安比较好的相亲网站?
怎么将XML数据可视化 D3.js加载XML
如何获取免费开源的自助建站系统源码?
公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?
Bpmn 2.0的XML文件怎么画流程图
微网站制作教程,我微信里的网站怎么才能复制到浏览器里?
网站制作培训多少钱一个月,网站优化seo培训课程有哪些?
如何高效完成自助建站业务培训?
浅析上传头像示例及其注意事项
陕西网站制作公司有哪些,陕西凌云电器有限公司官网?
宝塔建站助手安装配置与建站模板使用全流程解析
h5在线制作网站电脑版下载,h5网页制作软件?
如何在IIS中新建站点并配置端口与IP地址?
*请认真填写需求信息,我们会在24小时内与您取得联系。