1. 서문
처음 프로젝트를 개발할 당시에는 준비가 완벽하지 않았습니다. 개발이 특정 수준에 도달한 후에야 아직 해결되지 않은 문제가 있다는 것을 깨닫게 될 것입니다. 예를 들어, 오늘 저는 예외 처리라는 문제에 대해 이야기하고 싶습니다. 프로그램을 작성할 때 예외는 일반적으로 try...catch...finally를 통해 처리되는데, 프로그램을 작성할 때 실제로 가능한 모든 예외를 처리할 수 있을까요? 그리고 예외가 발생했을 때 어떤 로직이 실행되는지, 어떤 프롬프트 정보가 반환되는지, 어떤 페이지로 이동할지 등을 모두 고려해야 합니다.
2. @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>
3. AOP 기반 예외 처리
1. 컨트롤러 계층 예외 처리 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. 서비스 계층 예외 처리 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 { ......... }
4. 의심
@within(org.springframework.stereotype.Service), @Service
@annotation(org.springframework.web.bind.annotation. RequestMapping ), @RquestMapping
으로 주석이 달린 메서드를 가로챕니다. 5. 테스트
컨트롤러 계층과 서비스 계층에 대한 예외 테스트 클래스를 각각 작성합니다. 이것은 매우 간단합니다. 메서드에서 예외를 발생시키기만 하면 됩니다. 마지막으로 예외 발생 시 @AfterThrowing에 해당하는 메소드가 실행되는지 확인하면 된다.
위 내용은 이 글의 전체 내용입니다. 모든 분들의 학습에 도움이 되기를 바랍니다.
Spring Annotation AOP를 기반으로 한 더 많은 Java 기반 예외 처리 기사를 보려면 PHP 중국어 웹사이트를 주목하세요!