两个 servlet 中的 Spring、MVC 和 REST

Posted

技术标签:

【中文标题】两个 servlet 中的 Spring、MVC 和 REST【英文标题】:Spring, MVC and REST in two servlets 【发布时间】:2016-01-18 07:08:02 【问题描述】:

长话短说: 我有正确处理请求的 MVC servlet,但 REST servlet 似乎根本不处理任何请求。好像它不存在,或者请求不匹配。

更多详情:

我的spring版本是4.2.1。

我有一个带有 Spring MVC 的应用程序,它已经可以工作了。问题是我要添加的第二个 servlet,因此它可以处理 REST 请求。

这是web.xml 之前我尝试将 REST servlet 添加到其中(所以到目前为止一切正常):

<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
     xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee 
     http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
     version="3.1">

    <display-name>My app</display-name>

    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/core-context.xml /WEB-INF/spring-security.xml</param-value>
    </context-param>

    <listener>
        <listener-class>
            org.springframework.web.context.ContextLoaderListener
        </listener-class>
    </listener>

    <servlet-mapping>
        <servlet-name>AccServlet</servlet-name>
        <url-pattern>*.do</url-pattern>
    </servlet-mapping>

    <welcome-file-list>
        <welcome-file>index.do</welcome-file>
    </welcome-file-list>

    <servlet>
        <servlet-name>RestServlet</servlet-name>
        <servlet-class>
            org.springframework.web.servlet.DispatcherServlet
        </servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <filter>
        <filter-name>springSecurityFilterChain</filter-name>
        <filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
    </filter>

    <filter-mapping>
        <filter-name>springSecurityFilterChain</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>

    <session-config>
        <session-timeout>10</session-timeout>
    </session-config>
</web-app>

简而言之 - 所有对*.do 的请求都将由 MVC servlet 处理,所有请求通常都会通过 Spring Security。

我有一个简单的 MVC 控制器来处理它:

@Controller
public class JspController 

    @Autowired
    private UserService userService;

    @RequestMapping(value = "/index.do", method = RequestMethod.GET)
    public String provideIndexModel(ModelMap model) 
        User user = userService.getLoggedUser();
        model.addAttribute("user", user.getUsername());
        if (user.getBusiness() != null) 
            model.addAttribute("business", user.getBusiness().getCompanyName());
        
        return "index";
    

    @RequestMapping(value = "/login.do", method = RequestMethod.GET)
    public String provideLoginModel(ModelMap model) 
        return "login";
    

    public UserService getUserService() 
        return userService;
    

    public void setUserService(UserService userService) 
        this.userService = userService;
    

控制器位于通过 servlet 的上下文 xml 文件中的正确扫描声明扫描的包中。

现在,我启用 REST 的方法是将以下 servlet 添加到 web.xml

<servlet>
    <servlet-name>AccServlet</servlet-name>
    <servlet-class>
        org.springframework.web.servlet.DispatcherServlet
    </servlet-class>
    <load-on-startup>1</load-on-startup>
</servlet>

<servlet-mapping>
    <servlet-name>RestServlet</servlet-name>
    <url-pattern>*.rest</url-pattern>
</servlet-mapping>

然后我创建了另一个 servlet 上下文文件 RestServlet-servlet.xml,我在其中放置了 &lt;context:component-scan base-package="my.app.rest"/&gt; - 这就是包,然后我在其中放置了一个 RestController,如下所示:

@RestController
public class RestService 

    public RestService() 
        System.out.println("yes!");
    

    @RequestMapping(value = "/user.rest", method = RequestMethod.GET)
    public String getUser() 
        return "abc";
    

确实创建了其余控制器(我调试了构造函数),所以 spring 确实看到了它,但是当我尝试从 Postman(Chrome 扩展)调用它时,我得到 404 响应。我的请求是 URL:localhost:8080/acc/user.restacc 是我的应用名称)。

为什么我的匹配 *.rest 的呼叫没有被重定向到正确的控制器? (我试图在调试中停止 - 它从未被调用)。

编辑 1:

这是我当前的完整 web.xml:

<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee 
         http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
         version="3.1">

    <display-name>My app</display-name>

    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/core-context.xml /WEB-INF/spring-security.xml</param-value>
    </context-param>

    <listener>
        <listener-class>
            org.springframework.web.context.ContextLoaderListener
        </listener-class>
    </listener>

    <servlet-mapping>
        <servlet-name>AccServlet</servlet-name>
        <url-pattern>*.do</url-pattern>
    </servlet-mapping>

    <welcome-file-list>
        <welcome-file>index.do</welcome-file>
    </welcome-file-list>

    <servlet>
        <servlet-name>AccServlet</servlet-name>
        <servlet-class>
            org.springframework.web.servlet.DispatcherServlet
        </servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>RestServlet</servlet-name>
        <url-pattern>/rest/*</url-pattern>
    </servlet-mapping>

    <servlet>
        <servlet-name>RestServlet</servlet-name>
        <servlet-class>
            org.springframework.web.servlet.DispatcherServlet
        </servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <filter>
        <filter-name>springSecurityFilterChain</filter-name>
        <filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
    </filter>

    <filter-mapping>
        <filter-name>springSecurityFilterChain</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>

    <session-config>
        <session-timeout>10</session-timeout>
    </session-config>
</web-app>

这是我的 core-context.xml:

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd   
        http://www.springframework.org/schema/tx
        http://www.springframework.org/schema/tx/spring-tx.xsd
        http://www.springframework.org/schema/context     
        http://www.springframework.org/schema/context/spring-context.xsd">

    <context:component-scan base-package="my.app">
        <context:exclude-filter expression="org.springframework.stereotype.Controller"
            type="annotation" />
    </context:component-scan>

    <bean id="accDataSource" class="org.apache.commons.dbcp.BasicDataSource"
        destroy-method="close">
        <property name="driverClassName" value="org.postgresql.Driver" />
        <property name="url" value="jdbc:postgresql://localhost/acc" />
        <property name="username" value="acc" />
    </bean>

    <bean id="accSessionFactory"
        class="org.springframework.orm.hibernate5.LocalSessionFactoryBean">
        <property name="dataSource" ref="accDataSource" />
        <property name="packagesToScan">
            <list>
                <value>my.app.entities</value>
            </list>
        </property>
        <property name="hibernateProperties">
            <value>
                hibernate.dialect=org.hibernate.dialect.PostgreSQL92Dialect
            </value>
        </property>
    </bean>

    <bean id="txManager"
        class="org.springframework.orm.hibernate5.HibernateTransactionManager">
        <property name="sessionFactory" ref="accSessionFactory" />
    </bean>

    <tx:annotation-driven transaction-manager="txManager" />

</beans>

这是 AccServlet-servlet.xml:

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd     
        http://www.springframework.org/schema/context     
        http://www.springframework.org/schema/context/spring-context.xsd">

    <context:component-scan base-package="my.app.web"/>

    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
      <property name="prefix" value="/WEB-INF/jsp/" />
      <property name="suffix" value=".jsp" />
    </bean>

</beans>

这是我的 RestServlet-servlet.xml:

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd     
        http://www.springframework.org/schema/context     
        http://www.springframework.org/schema/context/spring-context.xsd">

    <context:component-scan base-package="my.app.rest"/>

</beans>

我在my.app.web 中有用于MVC 的@Controller,在my.app.rest 中有@RestController

来自 tomcat 的日志:

20-Oct-2015 14:00:16.096 INFO [main] org.apache.catalina.startup.VersionLoggerListener.log Server version:        Apache Tomcat/8.0.23
20-Oct-2015 14:00:16.097 INFO [main] org.apache.catalina.startup.VersionLoggerListener.log Server built:          May 19 2015 14:58:38 UTC
20-Oct-2015 14:00:16.097 INFO [main] org.apache.catalina.startup.VersionLoggerListener.log Server number:         8.0.23.0
20-Oct-2015 14:00:16.097 INFO [main] org.apache.catalina.startup.VersionLoggerListener.log OS Name:               Linux
20-Oct-2015 14:00:16.097 INFO [main] org.apache.catalina.startup.VersionLoggerListener.log OS Version:            3.14.33
20-Oct-2015 14:00:16.097 INFO [main] org.apache.catalina.startup.VersionLoggerListener.log Architecture:          amd64
20-Oct-2015 14:00:16.097 INFO [main] org.apache.catalina.startup.VersionLoggerListener.log Java Home:             /usr/lib64/java/jre
20-Oct-2015 14:00:16.097 INFO [main] org.apache.catalina.startup.VersionLoggerListener.log JVM Version:           1.8.0_45-b14
20-Oct-2015 14:00:16.097 INFO [main] org.apache.catalina.startup.VersionLoggerListener.log JVM Vendor:            Oracle Corporation
20-Oct-2015 14:00:16.098 INFO [main] org.apache.catalina.core.AprLifecycleListener.lifecycleEvent The APR based Apache Tomcat Native library which allows optimal performance in production environments was not found on the java.library.path: /usr/java/packages/lib/amd64:/usr/lib64:/lib64:/lib:/usr/lib
20-Oct-2015 14:00:16.174 INFO [main] org.apache.coyote.AbstractProtocol.init Initializing ProtocolHandler ["http-nio-8080"]
20-Oct-2015 14:00:16.183 INFO [main] org.apache.tomcat.util.net.NioselectorPool.getSharedSelector Using a shared selector for servlet write/read
20-Oct-2015 14:00:16.184 INFO [main] org.apache.coyote.AbstractProtocol.init Initializing ProtocolHandler ["ajp-nio-8009"]
20-Oct-2015 14:00:16.185 INFO [main] org.apache.tomcat.util.net.NioSelectorPool.getSharedSelector Using a shared selector for servlet write/read
20-Oct-2015 14:00:16.185 INFO [main] org.apache.catalina.startup.Catalina.load Initialization processed in 297 ms
20-Oct-2015 14:00:16.200 INFO [main] org.apache.catalina.core.StandardService.startInternal Starting service Catalina
20-Oct-2015 14:00:16.200 INFO [main] org.apache.catalina.core.StandardEngine.startInternal Starting Servlet Engine: Apache Tomcat/8.0.23
20-Oct-2015 14:00:16.213 INFO [localhost-startStop-1] org.apache.catalina.startup.HostConfig.deployWAR Deploying web application archive /home/googie/projects/tomcat/webapps/acc.war
20-Oct-2015 14:00:16.951 INFO [localhost-startStop-1] org.apache.jasper.servlet.TldScanner.scanJars At least one JAR was scanned for TLDs yet contained no TLDs. Enable debug logging for this logger for a complete list of JARs that were scanned but no TLDs were found in them. Skipping unneeded JARs during scanning can improve startup time and JSP compilation time.
20-Oct-2015 14:00:16.968 INFO [localhost-startStop-1] org.springframework.web.context.ContextLoader.initWebApplicationContext Root WebApplicationContext: initialization started
20-Oct-2015 14:00:17.012 INFO [localhost-startStop-1] org.springframework.web.context.support.XmlWebApplicationContext.prepareRefresh Refreshing Root WebApplicationContext: startup date [Tue Oct 20 14:00:17 CEST 2015]; root of context hierarchy
20-Oct-2015 14:00:17.031 INFO [localhost-startStop-1] org.springframework.beans.factory.xml.XmlBeanDefinitionReader.loadBeanDefinitions Loading XML bean definitions from ServletContext resource [/WEB-INF/core-context.xml]
20-Oct-2015 14:00:17.149 INFO [localhost-startStop-1] org.springframework.beans.factory.xml.XmlBeanDefinitionReader.loadBeanDefinitions Loading XML bean definitions from ServletContext resource [/WEB-INF/spring-security.xml]
20-Oct-2015 14:00:17.176 INFO [localhost-startStop-1] org.springframework.security.core.SpringSecurityCoreVersion.performVersionChecks You are running with Spring Security Core 4.0.2.RELEASE
20-Oct-2015 14:00:17.178 INFO [localhost-startStop-1] org.springframework.security.config.SecurityNamespaceHandler.<init> Spring Security 'config' module version is 4.0.2.RELEASE
20-Oct-2015 14:00:17.197 INFO [localhost-startStop-1] org.springframework.security.config.http.FilterInvocationSecurityMetadataSourceParser.parseInterceptUrlsForFilterInvocationRequestMap Creating access control expression attribute 'hasRole('ROLE_USER')' for /**
20-Oct-2015 14:00:17.231 INFO [localhost-startStop-1] org.springframework.security.config.http.HttpSecurityBeanDefinitionParser.checkFilterChainOrder Checking sorted filter chain: [Root bean: class [org.springframework.security.web.context.SecurityContextPersistenceFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 200, Root bean: class [org.springframework.security.web.context.request.async.WebAsyncManagerIntegrationFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 400, Root bean: class [org.springframework.security.web.header.HeaderWriterFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 500, Root bean: class [org.springframework.security.web.authentication.logout.LogoutFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 700, <org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter#0>, order = 1100, Root bean: class [org.springframework.security.web.authentication.www.BasicAuthenticationFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 1500, Root bean: class [org.springframework.security.web.savedrequest.RequestCacheAwareFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 1600, Root bean: class [org.springframework.security.web.servletapi.SecurityContextHolderAwareRequestFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 1700, Root bean: class [org.springframework.security.web.authentication.rememberme.RememberMeAuthenticationFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 1900, Root bean: class [org.springframework.security.web.authentication.AnonymousAuthenticationFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 2000, Root bean: class [org.springframework.security.web.session.SessionManagementFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 2100, Root bean: class [org.springframework.security.web.access.ExceptionTranslationFilter]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null, order = 2200, <org.springframework.security.web.access.intercept.FilterSecurityInterceptor#0>, order = 2300]
20-Oct-2015 14:00:17.489 INFO [localhost-startStop-1] org.hibernate.Version.logVersion HHH000412: Hibernate Core 5.0.2.Final
20-Oct-2015 14:00:17.490 INFO [localhost-startStop-1] org.hibernate.cfg.Environment.<clinit> HHH000206: hibernate.properties not found
20-Oct-2015 14:00:17.491 INFO [localhost-startStop-1] org.hibernate.cfg.Environment.buildBytecodeProvider HHH000021: Bytecode provider name : javassist
20-Oct-2015 14:00:17.521 INFO [localhost-startStop-1] org.hibernate.annotations.common.reflection.java.JavaReflectionManager.<clinit> HCANN000001: Hibernate Commons Annotations 5.0.0.Final
20-Oct-2015 14:00:17.613 INFO [localhost-startStop-1] org.hibernate.dialect.Dialect.<init> HHH000400: Using dialect: org.hibernate.dialect.PostgreSQL92Dialect
20-Oct-2015 14:00:17.704 INFO [localhost-startStop-1] org.hibernate.engine.jdbc.env.internal.LobCreatorBuilderImpl.useContextualLobCreation HHH000424: Disabling contextual LOB creation as createClob() method threw error : java.lang.reflect.InvocationTargetException
20-Oct-2015 14:00:17.705 INFO [localhost-startStop-1] org.hibernate.type.BasicTypeRegistry.register HHH000270: Type registration [java.util.UUID] overrides previous : org.hibernate.type.UUIDBinaryType@43d414e2
20-Oct-2015 14:00:18.117 INFO [localhost-startStop-1] org.springframework.orm.hibernate5.HibernateTransactionManager.afterPropertiesSet Using DataSource [org.apache.commons.dbcp.BasicDataSource@4c992a74] of Hibernate SessionFactory for HibernateTransactionManager
20-Oct-2015 14:00:18.130 INFO [localhost-startStop-1] org.springframework.security.web.DefaultSecurityFilterChain.<init> Creating filter chain: Ant [pattern='/lib/**'], []
20-Oct-2015 14:00:18.132 INFO [localhost-startStop-1] org.springframework.security.web.DefaultSecurityFilterChain.<init> Creating filter chain: Ant [pattern='/plugins/**'], []
20-Oct-2015 14:00:18.133 INFO [localhost-startStop-1] org.springframework.security.web.DefaultSecurityFilterChain.<init> Creating filter chain: Ant [pattern='/dist/**'], []
20-Oct-2015 14:00:18.134 INFO [localhost-startStop-1] org.springframework.security.web.DefaultSecurityFilterChain.<init> Creating filter chain: Ant [pattern='/bootstrap/**'], []
20-Oct-2015 14:00:18.135 INFO [localhost-startStop-1] org.springframework.security.web.DefaultSecurityFilterChain.<init> Creating filter chain: Ant [pattern='/css/**'], []
20-Oct-2015 14:00:18.136 INFO [localhost-startStop-1] org.springframework.security.web.DefaultSecurityFilterChain.<init> Creating filter chain: Ant [pattern='/fonts/**'], []
20-Oct-2015 14:00:18.137 INFO [localhost-startStop-1] org.springframework.security.web.DefaultSecurityFilterChain.<init> Creating filter chain: Ant [pattern='/login*'], []
20-Oct-2015 14:00:18.270 INFO [localhost-startStop-1] org.springframework.security.web.DefaultSecurityFilterChain.<init> Creating filter chain: org.springframework.security.web.util.matcher.AnyRequestMatcher@1, [org.springframework.security.web.context.SecurityContextPersistenceFilter@4a5281b8, org.springframework.security.web.context.request.async.WebAsyncManagerIntegrationFilter@42066dde, org.springframework.security.web.header.HeaderWriterFilter@7fda7a84, org.springframework.security.web.authentication.logout.LogoutFilter@67125593, org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter@39134788, org.springframework.security.web.authentication.www.BasicAuthenticationFilter@488ab5e9, org.springframework.security.web.savedrequest.RequestCacheAwareFilter@6428f826, org.springframework.security.web.servletapi.SecurityContextHolderAwareRequestFilter@116387fd, org.springframework.security.web.authentication.rememberme.RememberMeAuthenticationFilter@32efbd5e, org.springframework.security.web.authentication.AnonymousAuthenticationFilter@2f576317, org.springframework.security.web.session.SessionManagementFilter@c9d5713, org.springframework.security.web.access.ExceptionTranslationFilter@14c7512, org.springframework.security.web.access.intercept.FilterSecurityInterceptor@4e063d28]
20-Oct-2015 14:00:18.277 INFO [localhost-startStop-1] org.springframework.security.config.http.DefaultFilterChainValidator.checkLoginPageIsntProtected Checking whether login URL '/login.do' is accessible with your configuration
20-Oct-2015 14:00:18.297 INFO [localhost-startStop-1] org.springframework.web.context.ContextLoader.initWebApplicationContext Root WebApplicationContext: initialization completed in 1329 ms
20-Oct-2015 14:00:18.320 INFO [localhost-startStop-1] org.springframework.web.servlet.DispatcherServlet.initServletBean FrameworkServlet 'AccServlet': initialization started
20-Oct-2015 14:00:18.322 INFO [localhost-startStop-1] org.springframework.web.context.support.XmlWebApplicationContext.prepareRefresh Refreshing WebApplicationContext for namespace 'AccServlet-servlet': startup date [Tue Oct 20 14:00:18 CEST 2015]; parent: Root WebApplicationContext
20-Oct-2015 14:00:18.322 INFO [localhost-startStop-1] org.springframework.beans.factory.xml.XmlBeanDefinitionReader.loadBeanDefinitions Loading XML bean definitions from ServletContext resource [/WEB-INF/AccServlet-servlet.xml]
20-Oct-2015 14:00:18.369 INFO [localhost-startStop-1] org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping.registerHandler Mapped URL path [/index.do] onto handler 'jspController'
20-Oct-2015 14:00:18.369 INFO [localhost-startStop-1] org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping.registerHandler Mapped URL path [/login.do] onto handler 'jspController'
20-Oct-2015 14:00:18.517 INFO [localhost-startStop-1] org.springframework.web.servlet.DispatcherServlet.initServletBean FrameworkServlet 'AccServlet': initialization completed in 197 ms
20-Oct-2015 14:00:18.517 INFO [localhost-startStop-1] org.springframework.web.servlet.DispatcherServlet.initServletBean FrameworkServlet 'RestServlet': initialization started
20-Oct-2015 14:00:18.518 INFO [localhost-startStop-1] org.springframework.web.context.support.XmlWebApplicationContext.prepareRefresh Refreshing WebApplicationContext for namespace 'RestServlet-servlet': startup date [Tue Oct 20 14:00:18 CEST 2015]; parent: Root WebApplicationContext
20-Oct-2015 14:00:18.518 INFO [localhost-startStop-1] org.springframework.beans.factory.xml.XmlBeanDefinitionReader.loadBeanDefinitions Loading XML bean definitions from ServletContext resource [/WEB-INF/RestServlet-servlet.xml]
20-Oct-2015 14:00:18.539 INFO [localhost-startStop-1] org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping.registerHandler Mapped URL path [/rest/user] onto handler 'restService'
20-Oct-2015 14:00:18.539 INFO [localhost-startStop-1] org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping.registerHandler Mapped URL path [/rest/user.*] onto handler 'restService'
20-Oct-2015 14:00:18.539 INFO [localhost-startStop-1] org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping.registerHandler Mapped URL path [/rest/user/] onto handler 'restService'
20-Oct-2015 14:00:18.545 INFO [localhost-startStop-1] org.springframework.web.servlet.DispatcherServlet.initServletBean FrameworkServlet 'RestServlet': initialization completed in 28 ms
20-Oct-2015 14:00:18.552 INFO [localhost-startStop-1] org.apache.catalina.startup.HostConfig.deployWAR Deployment of web application archive /home/googie/projects/tomcat/webapps/acc.war has finished in 2,338 ms
20-Oct-2015 14:00:18.552 INFO [localhost-startStop-1] org.apache.catalina.startup.HostConfig.deployDirectory Deploying web application directory /home/googie/projects/tomcat/webapps/manager
20-Oct-2015 14:00:18.569 INFO [localhost-startStop-1] org.apache.catalina.startup.HostConfig.deployDirectory Deployment of web application directory /home/googie/projects/tomcat/webapps/manager has finished in 17 ms
20-Oct-2015 14:00:18.571 INFO [main] org.apache.coyote.AbstractProtocol.start Starting ProtocolHandler ["http-nio-8080"]
20-Oct-2015 14:00:18.578 INFO [main] org.apache.coyote.AbstractProtocol.start Starting ProtocolHandler ["ajp-nio-8009"]
20-Oct-2015 14:00:18.579 INFO [main] org.apache.catalina.startup.Catalina.start Server startup in 2394 ms
20-Oct-2015 14:00:37.639 INFO [http-nio-8080-exec-8] org.hibernate.hql.internal.QueryTranslatorFactoryInitiator.initiateService HHH000397: Using ASTQueryTranslatorFactory
20-Oct-2015 14:00:50.965 WARNING [http-nio-8080-exec-6] org.springframework.web.servlet.PageNotFound.noHandlerFound No mapping found for HTTP request with URI [/acc/rest/user] in DispatcherServlet with name 'RestServlet'

请注意,日志中的最后一行来自我的 GET 请求尝试到其余服务。

【问题讨论】:

您的 web.xml 是针对 Servlet 2.3 的,因此即使您的目标容器是较新版本,很多东西也很有可能以回退兼容性方式运行。去掉 doctype 并升级 &lt;web-app&gt; 声明以匹配目标容器支持的最新版本(Tomcat 6 = Servlet 2.5,Tomcat 7 = Servlet 3.0,Tomcat 8 = Servlet 3.1)然后重试。 好点。我已经修复它(到 3.1 版),但核心问题仍然存在。 当我升级到 servlet 3.1 时,一个有趣的事情开始出现在 tomcat 日志中。现在它在启动时打印:Mapped URL path [/acc/rest/user] onto handler 'restService',但后来当我调用我在日志中看到的其余路径时:org.springframework.web.servlet.PageNotFound.noHandlerFound No mapping found for HTTP request with URI [/acc/rest/user] in DispatcherServlet with name 'RestServlet'。我玩过几个映射(前面包含/acc,没有它等等 - 不幸的是没有任何效果)。 在映射中尝试 /*.rest/*。 我相信您应该在您的应用上下文 xml 中包含 &lt;mvc:annotation-driven&gt;,但似乎您可能错过了它。 【参考方案1】:

发现问题。事实证明,两个 *-servlet.xml 上下文文件都应该有 &lt;mvc:annotation-driven /&gt; 声明,以便控制器正常工作。没有它,servlet 可以工作,但映射没有正确处理。

【讨论】:

【参考方案2】:

在我看来,servlet 定义和映射(&lt;servlet-name&gt;RestServlet&lt;/servlet-name&gt;&lt;servlet-name&gt;AccServlet&lt;/servlet-name&gt;)中 servlet 名称的差异导致了问题。

您将调度程序 servlet 命名为 AccServlet 并且为 RestServlet 提供映射将不起作用。

在 spring-mvc 中,所有委托都由 DispatcherServlet 完成。可能的组合是:

<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>/WEB-INF/core-context.xml /WEB-INF/spring-security.xml</param-value>
</context-param>

<listener>
    <listener-class>
        org.springframework.web.context.ContextLoaderListener
    </listener-class>
</listener>
<servlet>
    <servlet-name>AccServlet</servlet-name>
    <servlet-class>
        org.springframework.web.servlet.DispatcherServlet
    </servlet-class>
    <load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
    <servlet-name>AccServlet</servlet-name>
    <url-pattern>/*</url-pattern>
</servlet-mapping>
<filter>
    <filter-name>springSecurityFilterChain</filter-name>
    <filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
</filter>

<filter-mapping>
    <filter-name>springSecurityFilterChain</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

<session-config>
    <session-timeout>10</session-timeout>
</session-config>

那么你的 REST 控制器可能是这样的:

@RestController
@RequestMapping("/rest")
public class RestService 

你所有的其他客户都会使用http://ip:port/yourapp/rest/something模式的URL MVC 控制器可以是:

@Controller
@RequestMapping("/mvc")
public class JspController 

  @RequestMapping(value = "/index.do", method = RequestMethod.GET)
  public String provideIndexModel(ModelMap model) 
      User user = userService.getLoggedUser();
      model.addAttribute("user", user.getUsername());
      if (user.getBusiness() != null) 
          model.addAttribute("business",   user.getBusiness().getCompanyName());
      
      return "index";
  

对 MVC 内容使用模式 localhost:8080/acc/mvc/index.do 的 URL。

【讨论】:

我已经尝试过了,但现在它们都不起作用。既不是mvc,也不是rest。两者都返回 404。我尝试打开 http://ip:port/yourapp/mvc/http://ip:port/yourapp/http://ip:port/yourapp/mvc/index.do - 没有成功。 而且我没有为单个 servlet 做 2 个映射 - 请注意我声明了 2 个 servlet,它们为每个 servlet 定义了一个映射。 我看不到这 2 个 servlet。你能分享统一的 web.xml 吗?此外,我更新了应该与我建议的配置一起使用的 URL。您使用的是 REST 客户端/工具吗? 另外,你能用日志语句更新你的问题吗? 我已经添加了您要求的所有信息(连同上下文文件)。关于两个 servlet - 我有两个定义为“AccServlet”和“RestServlet”的 servlet - 每个在 &lt;servlet&gt; 中都有自己的声明,在 &lt;servlet-mapping&gt; 中定义了自己的映射。

以上是关于两个 servlet 中的 Spring、MVC 和 REST的主要内容,如果未能解决你的问题,请参考以下文章

Java EE - Servlet 3.0 和 Spring MVC

servlet使用sessio和spring mvc中的controller使用session

Spring MVC中,applicationContext.xml [ServletName]-servlet.xml配置文件在web.xml中的配置详解...

Spring原理之MVC

Spring:MVC启动时的WebApplicationContext的关系

自己手撸一个 Spring MVC