Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/spring/11.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Java SpringREST应用程序配置和启动_Java_Spring_Maven 3_Spring Rest - Fatal编程技术网

Java SpringREST应用程序配置和启动

Java SpringREST应用程序配置和启动,java,spring,maven-3,spring-rest,Java,Spring,Maven 3,Spring Rest,我有这个应用程序,它过去工作得很好 我试着在很长一段时间后运行它,但它的行为并不像预期的那样 对于不太准确的错误问题,很抱歉 这是一个SpringREST应用程序,它公开了控制器,而不是SpringMVC应用程序 让我困惑的第一件事是,在调试器中启动它不会让它在WebConfiguration类的方法中放置的所有断点处停止: package com.nsn.nitro.project.rest.config; @Configuration @EnableWebMvc @EnableSpring

我有这个应用程序,它过去工作得很好

我试着在很长一段时间后运行它,但它的行为并不像预期的那样

对于不太准确的错误问题,很抱歉

这是一个SpringREST应用程序,它公开了控制器,而不是SpringMVC应用程序

让我困惑的第一件事是,在调试器中启动它不会让它在
WebConfiguration
类的方法中放置的所有断点处停止:

package com.nsn.nitro.project.rest.config;

@Configuration
@EnableWebMvc
@EnableSpringDataWebSupport
@ComponentScan(nameGenerator = PackageBeanNameGenerator.class, basePackages = { "com.nsn.nitro.project.rest" })
public class WebConfiguration extends WebMvcConfigurerAdapter {

    private static final int PAGE_DEFAULT_SIZE = 20;

    @Override
    public void addArgumentResolvers(
            List<HandlerMethodArgumentResolver> argumentResolvers) {
        PageableHandlerMethodArgumentResolver resolver = new PageableHandlerMethodArgumentResolver();
        resolver.setFallbackPageable(new PageRequest(0, PAGE_DEFAULT_SIZE));
        resolver.setMaxPageSize(50);
        resolver.setOneIndexedParameters(true);
        resolver.setPageParameterName("page");
        resolver.setSizeParameterName("size");
        argumentResolvers.add(resolver);
        super.addArgumentResolvers(argumentResolvers);
    }

    @Bean
    public MessageSource messageSource() {
        ReloadableResourceBundleMessageSource messageSource = new ReloadableResourceBundleMessageSource();
        messageSource.setBasenames("classpath:messages/messages",
                "classpath:messages/validation");
        // If true, the key of the message will be displayed if the key is not
        // found, instead of throwing an exception
        messageSource.setUseCodeAsDefaultMessage(true);
        messageSource.setDefaultEncoding("UTF-8");
        // The value 0 means always reload the messages to be developer friendly
        messageSource.setCacheSeconds(0);
        return messageSource;
    }

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

    @Override
    public void configureDefaultServletHandling(
            DefaultServletHandlerConfigurer configurer) {
        configurer.enable();
    }

    @Bean
    public LocaleResolver localeResolver() {
        return new SmartLocaleResolver();
    }

    @Bean
    public MultipartResolver multipartResolver() {
        CommonsMultipartResolver multipartResolver = new CommonsMultipartResolver();
        multipartResolver.setMaxUploadSize(1024000);
        return multipartResolver;
    }

    // The locale interceptor provides a way to switch the language in any page
    // just by passing the lang=’en’, lang=’fr’, and so on to the url
    @Bean
    public LocaleChangeInterceptor localeChangeInterceptor() {
        LocaleChangeInterceptor localeChangeInterceptor = new LocaleChangeInterceptor();
        localeChangeInterceptor.setParamName("lang");
        return localeChangeInterceptor;
    }

    // Avoid caching issue with AngularJS on IE
    @Bean
    public WebContentInterceptor webContentInterceptor() {
        WebContentInterceptor interceptor = new WebContentInterceptor();
        interceptor.setCacheSeconds(0);
        interceptor.setUseExpiresHeader(true);
        interceptor.setUseCacheControlHeader(true);
        interceptor.setUseCacheControlNoStore(true);
        return interceptor;
    }

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(localeChangeInterceptor());
        registry.addInterceptor(webContentInterceptor());
    }
调试器也不会在ApplicationConfiguration()构造函数中的断点处停止。我在控制台日志中看不到记录器输出

但应用程序似乎启动良好:

INFO] <<< tomcat7-maven-plugin:2.2:run (default-cli) < process-classes @ nitro-project-rest <<<
[INFO] 
[INFO] --- tomcat7-maven-plugin:2.2:run (default-cli) @ nitro-project-rest ---
[INFO] Running war on http://localhost:8080/nitro-project-rest
[INFO] Creating Tomcat server configuration at /home/stephane/dev/java/projects/nitro-project-rest/target/tomcat
[INFO] create webapp with contextPath: /nitro-project-rest
Dec 31, 2016 9:41:14 AM org.apache.coyote.AbstractProtocol init
INFO: Initializing ProtocolHandler ["http-bio-8080"]
Dec 31, 2016 9:41:14 AM org.apache.coyote.AbstractProtocol init
INFO: Initializing ProtocolHandler ["http-bio-8443"]
Dec 31, 2016 9:41:16 AM org.apache.catalina.core.StandardService startInternal
INFO: Starting service Tomcat
Dec 31, 2016 9:41:16 AM org.apache.catalina.core.StandardEngine startInternal
INFO: Starting Servlet Engine: Apache Tomcat/7.0.47
Dec 31, 2016 9:41:25 AM org.apache.catalina.core.ApplicationContext log
INFO: 1 Spring WebApplicationInitializers detected on classpath
SLF4J: Class path contains multiple SLF4J bindings.
SLF4J: Found binding in [jar:file:/home/stephane/.m2/repository/org/slf4j/slf4j-log4j12/1.6.4/slf4j-log4j12-1.6.4.jar!/org/slf4j/impl/StaticLoggerBinder.class]
SLF4J: Found binding in [jar:file:/home/stephane/.m2/repository/ch/qos/logback/logback-classic/1.1.7/logback-classic-1.1.7.jar!/org/slf4j/impl/StaticLoggerBinder.class]
SLF4J: See http://www.slf4j.org/codes.html#multiple_bindings for an explanation.
SLF4J: Actual binding is of type [ch.qos.logback.classic.util.ContextSelectorStaticBinder]
Dec 31, 2016 9:41:26 AM org.apache.coyote.AbstractProtocol start
INFO: Starting ProtocolHandler ["http-bio-8080"]
Dec 31, 2016 9:41:26 AM org.apache.coyote.AbstractProtocol start
INFO: Starting ProtocolHandler ["http-bio-8443"]
但是发送任何请求,比如
curl-H“Accept:application/json”--用户nsn@nsn.com:******* http://localhost:8080/nitro-project rest/greeting?message=Steph
总是给我以下错误:

SEVERE: Servlet.service() for servlet [default] in context with path [/nitro-project-rest] threw exception
java.lang.IllegalStateException: No WebApplicationContext found: no ContextLoaderListener or DispatcherServlet registered?
这是我的问候控制器:

@Controller
@RequestMapping(RESTConstants.SLASH + RESTConstants.GREETING + RESTConstants.SLASH)
public class GreetingController {

    private static final String TEMPLATE = "Hello, %s!";

    @RequestMapping(value = RESTConstants.SLASH + RESTConstants.HELLO, method = RequestMethod.GET, produces = MediaType.APPLICATION_JSON_VALUE)
    @ResponseBody
    public HttpEntity<Greeting> hello(@RequestParam(value = "message", required = false, defaultValue = "World") String message, UriComponentsBuilder builder) {
        Greeting greeting = new Greeting(String.format(TEMPLATE, message));
        greeting.add(linkTo(methodOn(GreetingController.class).hello(message, builder)).withSelfRel());
        HttpHeaders responseHeaders = new HttpHeaders();
        responseHeaders.setLocation(builder.path(RESTConstants.SLASH + RESTConstants.GREETING + RESTConstants.SLASH + "hello").buildAndExpand().toUri());
        responseHeaders.add("Content-Type", "application/json; charset=utf-8");
        ResponseEntity<Greeting> responseEntity = new ResponseEntity<Greeting>(greeting, responseHeaders, HttpStatus.OK);        
        return responseEntity;
    }

}
@控制器
@RequestMapping(RESTConstants.SLASH+RESTConstants.GREETING+RESTConstants.SLASH)
公共类迎宾控制器{
私有静态最终字符串模板=“您好,%s!”;
@RequestMapping(value=restctants.SLASH+restctants.HELLO,method=RequestMethod.GET,products=MediaType.APPLICATION\u JSON\u value)
@应答器
公共HttpEntity hello(@RequestParam(value=“message”,required=false,defaultValue=“World”)字符串消息,UriComponentsBuilder){
问候语=新问候语(String.format(模板、消息));
add(linkTo(methodOn(GreetingController.class).hello(message,builder)).withSelfRel());
HttpHeaders responseHeaders=新的HttpHeaders();
setLocation(builder.path(restcontents.SLASH+restcontents.GREETING+restcontents.SLASH+hello”).buildAndExpand().toUri();
add(“内容类型”,“应用程序/json;字符集=utf-8”);
ResponseEntity ResponseEntity=新的ResponseEntity(问候语、响应负责人、HttpStatus.OK);
返回响应性;
}
}

我错过了dispatcher servlet

现在有了这个文件,它工作得更好了:

public class WebInit extends AbstractAnnotationConfigDispatcherServletInitializer {

    private static Logger logger = LoggerFactory.getLogger(WebInit.class);

    @Override
    protected void customizeRegistration(ServletRegistration.Dynamic registration) {
        registration.setInitParameter("dispatchOptionsRequest", "true");
        registration.setAsyncSupported(true);
        registration.setLoadOnStartup(1);
    }

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

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

    @Override
    protected String[] getServletMappings() {
        // The '/api/*' prefix is to allow a match of an .../api/... request to a controller mapping
        // The '/' is to allow a match of a root request to a static resource
        return new String[] { "/api/*", "/" };
    }

    @Override
    protected String getServletName() {
        return CommonConstants.SERVLET_NAME;
    }

}
公共类WebInit扩展了AbstractAnnotationConfigDispatcherServletInitializer{
私有静态记录器Logger=LoggerFactory.getLogger(WebInit.class);
@凌驾
受保护的无效自定义注册(ServletRegistration.Dynamic registration){
registration.setInitParameter(“DispatchOptions请求”、“true”);
registration.setAsyncSupported(true);
注册。设置加载启动(1);
}
@凌驾
受保护类[]getRootConfigClasses(){
返回新类[]{ApplicationConfiguration.Class};
}
@凌驾
受保护类[]getServletConfigClasses(){
返回新类[]{WebConfiguration.Class};
}
@凌驾
受保护的字符串[]getServletMappings(){
//“/api/*”前缀允许将…/api/…请求与控制器映射匹配
//“/”允许根请求与静态资源匹配
返回新字符串[]{“/api/*”,“/”};
}
@凌驾
受保护的字符串getServletName(){
返回CommonConstants.SERVLET_名称;
}
}

我错过了dispatcher servlet

现在有了这个文件,它工作得更好了:

public class WebInit extends AbstractAnnotationConfigDispatcherServletInitializer {

    private static Logger logger = LoggerFactory.getLogger(WebInit.class);

    @Override
    protected void customizeRegistration(ServletRegistration.Dynamic registration) {
        registration.setInitParameter("dispatchOptionsRequest", "true");
        registration.setAsyncSupported(true);
        registration.setLoadOnStartup(1);
    }

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

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

    @Override
    protected String[] getServletMappings() {
        // The '/api/*' prefix is to allow a match of an .../api/... request to a controller mapping
        // The '/' is to allow a match of a root request to a static resource
        return new String[] { "/api/*", "/" };
    }

    @Override
    protected String getServletName() {
        return CommonConstants.SERVLET_NAME;
    }

}
公共类WebInit扩展了AbstractAnnotationConfigDispatcherServletInitializer{
私有静态记录器Logger=LoggerFactory.getLogger(WebInit.class);
@凌驾
受保护的无效自定义注册(ServletRegistration.Dynamic registration){
registration.setInitParameter(“DispatchOptions请求”、“true”);
registration.setAsyncSupported(true);
注册。设置加载启动(1);
}
@凌驾
受保护类[]getRootConfigClasses(){
返回新类[]{ApplicationConfiguration.Class};
}
@凌驾
受保护类[]getServletConfigClasses(){
返回新类[]{WebConfiguration.Class};
}
@凌驾
受保护的字符串[]getServletMappings(){
//“/api/*”前缀允许将…/api/…请求与控制器映射匹配
//“/”允许根请求与静态资源匹配
返回新字符串[]{“/api/*”,“/”};
}
@凌驾
受保护的字符串getServletName(){
返回CommonConstants.SERVLET_名称;
}
}

您的程序是否扫描路径以查找注释为@Configuration的WebConfiguration.java?我还添加了应用程序配置bean。但我想知道实际的扫描发生了什么。没有显示断点或记录器输出…看起来您的配置bean在运行时不存在,所以我需要更改我的问题,例如,您的程序是否扫描路径以查找“根配置文件”(似乎是应用程序配置)。我想您需要在网络配置中设置应用程序配置。@Bolzano在您之前的评论之后,我开始问自己这个问题。就在不久前,我处理了Spring,我忘记了这个配置“根”路径是在哪里指定的。您的程序是否扫描路径以查找注释为@Configuration的WebConfiguration.java?我还添加了应用程序配置bean。但我想知道实际的扫描发生了什么。没有显示断点或记录器输出…看起来您的配置bean在运行时不存在,所以我需要更改我的问题,例如,您的程序是否扫描路径以查找“根配置文件”(似乎是应用程序配置)。我想您需要在网络配置中设置应用程序配置。@Bolzano在您之前的评论之后,我开始问自己这个问题。就在不久前,我处理了Spring,我有点忘记了这个配置“根”路径是在哪里指定的。
public class WebInit extends AbstractAnnotationConfigDispatcherServletInitializer {

    private static Logger logger = LoggerFactory.getLogger(WebInit.class);

    @Override
    protected void customizeRegistration(ServletRegistration.Dynamic registration) {
        registration.setInitParameter("dispatchOptionsRequest", "true");
        registration.setAsyncSupported(true);
        registration.setLoadOnStartup(1);
    }

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

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

    @Override
    protected String[] getServletMappings() {
        // The '/api/*' prefix is to allow a match of an .../api/... request to a controller mapping
        // The '/' is to allow a match of a root request to a static resource
        return new String[] { "/api/*", "/" };
    }

    @Override
    protected String getServletName() {
        return CommonConstants.SERVLET_NAME;
    }

}