统一处理异常

image.png
使用 @ControllerAdvice 注解无需再任何Controller上做处理,spring既可以统一处理异常

  1. /**
  2. * 处理服务端异常(500)
  3. */
  4. //扫描带有 @controller 的组件
  5. @ControllerAdvice(annotations = Controller.class)
  6. public class ExceptionAdvice {
  7. private static final Logger logger = LoggerFactory.getLogger(ExceptionAdvice.class);
  8. @ExceptionHandler({Exception.class})
  9. public void handlerException(Exception e, HttpServletRequest request, HttpServletResponse response) throws IOException {
  10. logger.error("服务器发生异常:" + e.getMessage());
  11. for (StackTraceElement element:e.getStackTrace()){
  12. logger.error(element.toString());
  13. }
  14. //区分异步请求和普通请求
  15. String xRequestedWith = request.getHeader("x-requested-with");
  16. if("XMLHttpRequest".equals(xRequestedWith)){
  17. //异步请求(希望返回JSON数据)
  18. response.setContentType("application/plain;charset=utf-8");
  19. PrintWriter writer = response.getWriter();
  20. writer.write(CommunityUtil.getJSONString(1,"服务器异常"));
  21. }
  22. else{
  23. //普通请求(希望返回的是一个网页)
  24. response.sendRedirect(request.getContextPath() + "/error");
  25. }
  26. }

统一记录日志

image.png
image.png
使用AOP对系统各个业务模块统一做处理,例如,权限检查、记录日志、事务管理
image.png
已经开发好的Bean称为目标对象(Target) 目标对象上有很多地点可以织入代码称为连接点(Joinpoint)
AOP解决这些统一需求的方式是将代码定义到一个额外的Bean 称为方面组件(Aspect),这些方面组件会在程序运行或之前被框架织入到某些连接点之上,方面组件里通过切点(pointcut)声明方面组件要被织入到哪些目标对象和连接点之上,通过通知(Advice)声明方面组件到底要处理什么样的逻辑
image.png
image.png

  1. /**
  2. * 统一日志管理
  3. */
  4. @Component
  5. @Aspect
  6. public class ServiceLogAspect {
  7. private static final Logger logger = LoggerFactory.getLogger(ServiceLogAspect.class);
  8. //service包下所有的类所有的方法所有的参数所有的返回值都要做处理
  9. @Pointcut("execution(* com.voyager.community.service.*.*(..))")
  10. public void pointcut() {
  11. }
  12. @Before("pointcut()")
  13. public void before(JoinPoint joinPoint) {
  14. // 用户[IP 地址], 在某个时间访问了 [com.voyager.community.service.xxx]
  15. ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
  16. if (attributes == null) {
  17. return ;
  18. }
  19. HttpServletRequest request = attributes.getRequest();
  20. String ip = request.getRemoteHost();
  21. String time = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date());
  22. String target = joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName();
  23. logger.info(String.format("用户[%s], 在[%s], 访问了[%s].", ip, time, target));
  24. }
  25. }