【问题标题】:Spring @ControllerAdvice Handler exception is not picking NoHandlerFoundExceptionSpring @ControllerAdvice 处理程序异常未选择 NoHandlerFoundException
【发布时间】:2018-01-15 21:58:01
【问题描述】:

我正在使用@ControllerAdvice 处理异常,我有一个用于我的休息控制器的@ControllerAdvice 和一个用于非休息控制器的,他们正在为 Exception.class 选择异常,但他们没有选择 NoHandlerFoundException.class。

我们正在使用 Spring 4.3.7.RELEASE、Java Config、Apache 8.5.12。

下面是 MyWebInitializer 类

public class MyWebInitializer extends
        AbstractAnnotationConfigDispatcherServletInitializer {


    @Override
    public void onStartup(ServletContext servletContext) throws ServletException {      

        super.onStartup(servletContext);

        servletContext.addListener(new RequestContextListener());
        servletContext.addListener(new SpringSessionListener());

        ServletRegistration.Dynamic dispatcher1 =
        servletContext.addServlet("blackwellsjspservlet", new JspServlet());
        dispatcher1.setLoadOnStartup(3);
        dispatcher1.addMapping("*.do");
    }

    @Override
    protected Class<?>[] getRootConfigClasses() {
        return new Class[] { SpringRootConfig.class };
    }

    @Override
    protected Class<?>[] getServletConfigClasses() {
        return new Class[] { SpringWebConfig.class };
    }

    @Override
    protected String[] getServletMappings() {
        return new String[] { "/" };
    }

    @Override
      protected String getServletName() {
        return "blackwellsmvcdispatcherservlet";
      }

    //Added to allow customised NoHandlerFoundPage
    @Override
    protected void customizeRegistration(ServletRegistration.Dynamic registration) {
        boolean done = registration.setInitParameter("throwExceptionIfNoHandlerFound", "true"); // -> true
        if(!done) throw new RuntimeException();
    }

    //Added to register a default profile
    @Override
    protected WebApplicationContext createServletApplicationContext() {
        WebApplicationContext context = (WebApplicationContext) super.createServletApplicationContext();
        ((ConfigurableEnvironment) context.getEnvironment()).setDefaultProfiles("production");

        return context;
    }
}

我的 SpringWebConfig 是:

@EnableWebMvc
@Configuration
public class SpringWebConfig extends WebMvcConfigurerAdapter {

    @Autowired
    private ServletContext servletContext;

    @Autowired
    Environment env;

    @Override
    public void addResourceHandlers(ResourceHandlerRegistry registry) {
        registry.addResourceHandler("/resources/**").addResourceLocations("/resources/");
    }

    @Bean
    public InternalResourceViewResolver jspViewResolver() {
        InternalResourceViewResolver viewResolver = new InternalResourceViewResolver();
        viewResolver.setViewClass(JstlView.class);
        viewResolver.setPrefix("/");
        viewResolver.setViewNames("*.jsp");
        return viewResolver;
    }


    @Bean
    public Loader<?> templateLoader(){
        return new ServletLoader(servletContext);
    }

    @Bean
    public CustomSpringExtension springExtension() {
        return new CustomSpringExtension();
    }



    @Bean
    public PebbleEngine pebbleEngine() {
        return new PebbleEngine.Builder()
                .loader(this.templateLoader()).cacheActive(false)
                .extension(springExtension())
                .build();
    }

    @Bean
    public ViewResolver pebbleViewResolver() {
        PebbleViewResolver viewResolver = new PebbleViewResolver();
        viewResolver.setPrefix("/WEB-INF/patternLab/");
        viewResolver.setPebbleEngine(pebbleEngine());
        viewResolver.setOrder(Ordered.LOWEST_PRECEDENCE - 5);
        viewResolver.setViewNames("*.html");
        return viewResolver;
    }

    /**
     * Bean for ResourceMessageBundle
     * basename and folder resources/locale/messages. i.g. messages_en_GB.properties,messages_en_US.properties
     * @author Henrique Droog
     * @return
     */
    @Bean
    public MessageSource messageSource(){
        ReloadableResourceBundleMessageSource messageSource = new ReloadableResourceBundleMessageSource();
        messageSource.setBasename("classpath:locale/messages");
        messageSource.setDefaultEncoding("UTF-8");

        return messageSource;
    }

    /**
     * Bean for LocalResolver
     * set en as a default locale
     * @author Henrique Droog
     * @date 09.03.2017
     * @return
     */
    @Bean
    public LocaleResolver localeResolver(){
        CookieLocaleResolver resolver = new CookieLocaleResolver();
        resolver.setDefaultLocale(new Locale("en"));
        resolver.setCookieName("myLocaleCookie");
        resolver.setCookieMaxAge(4800);
        return resolver;
    }

    @Bean
    public MethodValidationPostProcessor methodValidationPostProcessor(){
        return new MethodValidationPostProcessor();
    }


    @Profile({"production-api","dev-api"})
        @ComponentScan(basePackages  = { "uk.co.blackwells.controllers", "uk.co.blackwells.rest.controller"})
    public static class ProductionAPIWebConfig {

    }

    @Profile({"production","dev","default"})
    @ComponentScan(basePackages  = { "uk.co.blackwells.controllers","uk.co.blackwells.rest.controller"})
    public static class ProductionWebConfig {

    }

    @Profile({"test"})
    @ComponentScan(basePackages  = { "uk.co.blackwells.controllers","uk.co.blackwells.rest.controller"})
    public static class TestWebConfig {

    }

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(sessionInterceptor()).addPathPatterns("/**").excludePathPatterns("/api/**", "/restful/**");

        LocaleChangeInterceptor interceptor = new LocaleChangeInterceptor();
        interceptor.setParamName("mylocale");
        registry.addInterceptor(interceptor);

    }

    @Bean
    public SessionInterceptor sessionInterceptor() {
        return new SessionInterceptor();
    }

    @Bean
    public Validator validatorFactory () {
        return new LocalValidatorFactoryBean();
    }
}

我的控制器对非休息控制器的建议:

@ControllerAdvice(annotations=Controller.class)
public class GlobalDefaultExceptionHandlerController {

    private final Logger logger = LoggerFactory.getLogger(GlobalDefaultExceptionHandlerController.class);
    ErrorList errorList;

    /**
     * This init binder register a custom editor to trim all fields coming
     * from html
     * @param binder
     */
    @InitBinder
    public void initBinder ( WebDataBinder binder )
    {
        StringTrimmerEditor stringtrimmer = new StringTrimmerEditor(true);
        binder.registerCustomEditor(String.class, stringtrimmer);
    }

    @ExceptionHandler(NoHandlerFoundException.class)
    public ModelAndView handleNotFoundException(HttpServletRequest req, NoHandlerFoundException ex) {

        ErrorMessage erMsg = new ErrorMessage();
        erMsg.setError(ex.getMessage());
        erMsg.setStackTrace(stackTrace);
        erMsg.setException(ex);

        ModelAndView model = new ModelAndView();
        model.addObject("errorMessageObject", erMsg);

        model.setViewName("pages/error-404.html");

        return model;
    }

    @ExceptionHandler(Exception.class)
    public ModelAndView handleAllException(HttpServletRequest req, Exception ex) throws Exception {

        // Rethrow annotated exceptions or they will be processed here instead.
        if (AnnotationUtils.findAnnotation(ex.getClass(), ResponseStatus.class) != null)
            throw ex;


        ErrorMessage erMsg = new ErrorMessage();
        erMsg.setError(exceptionCode);
        erMsg.setStackTrace(stackTrace);
        erMsg.setException(ex);

        ModelAndView model = new ModelAndView();
        model.addObject("errorMessageObject", erMsg);
        model.addObject("errMsg", ex.getMessage());
        model.addObject("exceptionCode", exceptionCode);

        model.setViewName("pages/error-500.html");

        return model;

    }

    @RequestMapping(value = {"/SessionError"}, method = RequestMethod.GET)
    public String sessionError() {
        return "pages/error-500.html";
    }
}

对于休息控制器:

@RestControllerAdvice(annotations=RestController.class)
public class ApiExceptionHandlerController {

    private final Logger logger = LoggerFactory.getLogger(ApiExceptionHandlerController.class);

    @ExceptionHandler(Exception.class)
    public ResponseEntity<Object> handleException(HttpServletRequest req, Exception ex){
        logger.error("API Exception: {}", ex.getMessage(), ex);
        logger.error("Path:{}, IP:{}", req.getServletPath(),  req.getRemoteAddr());

         ApiError apiError = 
                  new ApiError(HttpStatus.INTERNAL_SERVER_ERROR, "Internal server error", ex.getMessage());
        return new ResponseEntity<Object>(apiError, new HttpHeaders(), apiError.getStatus());
    }

    @ExceptionHandler(NoHandlerFoundException.class)
    public ResponseEntity<Object> handleNotFoundException(HttpServletRequest req, Exception ex){
        logger.error("API NoHandlerFoundException: {}", ex.getMessage(), ex);
        logger.error("Path:{}, IP:{}", req.getServletPath(),  req.getRemoteAddr());
        ex.printStackTrace();

         ApiError apiError = 
                  new ApiError(HttpStatus.NOT_FOUND, ex.getMessage(), ex.getMessage());
        return new ResponseEntity<Object>(apiError, new HttpHeaders(), apiError.getStatus());
    }
}

两个控制器建议都选择 NoHandlerFoundException 直到某个时候。我查看了这些文件的历史更改,但它们已经有一段时间没有更改了。我想也许这部分代码

@Override
    protected void customizeRegistration(ServletRegistration.Dynamic registration) {
        boolean done = registration.setInitParameter("throwExceptionIfNoHandlerFound", "true"); // -> true
        if(!done) throw new RuntimeException();
    }

正在被其他代码覆盖,但我找不到与此相关的任何内容。

【问题讨论】:

    标签: java spring spring-mvc exception-handling http-status-code-404


    【解决方案1】:

    如果你使用了 Spring Boot:

    在 org.springframework.web.servlet.DispatcherServlet 类中有一个名为 throwExceptionIfNoHandlerFound 的变量。如果设置为 true,则名为 noHandlerFound 的方法将抛出 NoHandlerFoundException。您的异常处理程序现在将捕获它。

    将此属性添加到您的 application.properties 文件中:spring.mvc.throw-exception-if-no-handler-found=true

    【讨论】:

    • 谢谢 Dave,很遗憾我没有使用 Spring Boot。我还在扩展 AbstractAnnotationConfigDispatcherServletInitializer 的 y Web 初始化程序中将 throwExceptionIfNoHandlerFound 设置为 true。无论如何,我会考虑是否可以尝试使用一些属性而不是网络初始化程序。
    猜你喜欢
    • 2016-03-25
    • 2019-06-04
    • 1970-01-01
    • 1970-01-01
    • 2018-07-13
    • 2017-11-23
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多