两个 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
,我在其中放置了 <context:component-scan base-package="my.app.rest"/>
- 这就是包,然后我在其中放置了一个 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.rest
(acc
是我的应用名称)。
为什么我的匹配 *.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 并升级<web-app>
声明以匹配目标容器支持的最新版本(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 中包含 <mvc:annotation-driven>
,但似乎您可能错过了它。
【参考方案1】:
发现问题。事实证明,两个 *-servlet.xml 上下文文件都应该有 <mvc:annotation-driven />
声明,以便控制器正常工作。没有它,servlet 可以工作,但映射没有正确处理。
【讨论】:
【参考方案2】:在我看来,servlet 定义和映射(<servlet-name>RestServlet</servlet-name>
和 <servlet-name>AccServlet</servlet-name>
)中 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 - 每个在 <servlet>
中都有自己的声明,在 <servlet-mapping>
中定义了自己的映射。以上是关于两个 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中的配置详解...