Spring @ControllerAdvice Handler异常没有选择NoHandlerFoundException

时间:2017-08-08 10:58:16

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

我正在使用@ControllerAdvice处理异常,我在@ControllerAdvice上为我的其他控制器和一个非rest控制器处理异常,他们正在为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();
    }
某些其他代码正在覆盖

,但我找不到与此相关的任何内容。

1 个答案:

答案 0 :(得分:1)

如果你使用Spring Boot:

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

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