我通过将Spring 3集成到一个传统的Servlet应用程序中并逐渐转换旧版应用程序来学习Spring。
类似于我正在使用的web.xml * -servlet.xml发布在下面。基本上设置了一些东西,以便检索像“搜索”这样的字符串,Spring会将它路由到Controller,而视图解析器会将“搜索”转换为“/jsp/search.jsp”
我遇到了从传统Servlet和传统ServletFilter到Spring执行响应 .sendRedirect(“搜索”)的问题。 URL正确显示,但我得到一个空白页,尽管System.out.println()调用表明已到达JSP。没有来自Spring和浏览器的错误消息只告诉我重定向出了问题。
我通过转发解决了这个问题,而不是从传统的Servlet和ServletFilters重定向:
request.getRequestDispatcher("search").forward(request,response);
getServletConfig().getServletContext().getRequestDispatcher("search").forward(request,response);
从Spring 3中完成的新JSP转向另一个方向,我在屏幕上有按钮,所以我只是使用了一个javascript调用“location.href = / helloworld”。我需要发送一些参数,所以我可能会将这些按钮转换为提交微小的HTML表单。
我想知道的是,是否有更好的方法让Spring 3和Legacy Servlet以更好的方式进行通信。
Legacy Servlet =>春天3
和
春天3 =>旧版Servlet
的web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app id="WebApp_ID" version="3.0"
xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd">
<display-name>Acme</display-name>
<!--welcome-file-list>
<welcome-file>/login</welcome-file>
</welcome-file-list-->
<servlet>
<servlet-name>acme</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>acme</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
<!-- Help Find The Spring Config Files -->
<listener>
<listener-class>
org.springframework.web.context.ContextLoaderListener
</listener-class>
</listener>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
/WEB-INF/nsd-servlet.xml,
/WEB-INF/nsd-security.xml
</param-value>
</context-param>
<!-- Spring Security -->
<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>
<!-- Integrate A Legacy Screen Done With A Servlet -->
<servlet>
<servlet-name>HelloWorldServlet</servlet-name>
<servlet-class>
com.legacy.HelloWorldServlet
</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>HelloWorldServlet</servlet-name>
<url-pattern>/helloworldservlet</url-pattern>
</servlet-mapping>
</web-app>
我的acme-servlet.xml
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd">
<context:component-scan base-package="com.acme.controller" />
<mvc:resources mapping = "/**" location = "/,file:/apps1/bea/user_projects/domains/acme/common/,file:/c:/weblogic_common_files/acme/"/>
<mvc:annotation-driven/>
<bean
class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name = "prefix" value = "/jsp/"/>
<property name = "suffix" value = ".jsp"/>
</bean>
</beans>
答案 0 :(得分:1)
当您使用response.sendRedirect(String url)
时,您实际上是在传递要重定向到的网址。
sendRedirect遵守以下规则:
/*
* Destination, it can be any relative or context specific path.
* if the path starts without '/' it is interpreted as relative to the current request URI.
* if the path starts with '/' it is interpreted as relative to the context.
*/
String destination ="/jsp/destination.jsp";
response.sendRedirect(response.encodeRedirectURL(destination));
当您输入“搜索”时,这与当前请求URI相关。因此,除非您当前的请求是/ acme / something,并且假设/ acme / search是您的servlet,否则请求将失败。
但是,如果您使用/ acme / search将路径相对于根目录放置,那么请求将适用于任何上下文。
话虽如此,我不相信这是最好的方法,因为重定向涉及将响应发送回客户端浏览器,告诉它再次从另一个URL获取内容。这似乎是一次往返于服务器的浪费之旅。
更好的方法可能是使用Spring MVC之类的东西将servlet连接到Spring。它非常灵活,您可以围绕现有的servlet包装新的控制器类,然后通过传入HttpServletRequest和HttpServletResponse对象直接调用它们。完成后,您可以使用工作,高效的系统慢慢消除任何冗余,而不是将其与重定向链接在一起。
package samples;
public class SampleController extends AbstractController {
private int cacheSeconds;
// for wiring in cacheSeconds
public void setCacheSeconds(int cacheSeconds) {
this.cacheSeconds = cacheSeconds;
}
public int getCacheSeconds() {
return cacheSeconds;
}
public ModelAndView handleRequestInternal(
HttpServletRequest request,
HttpServletResponse response) throws Exception {
// you now have a Spring Controller wired in, and you could delegate to
// your legacy servlet in this manner
YourServlet servlet = new YourServlet();
servlet.doGet(request, response);
ModelAndView mav = new ModelAndView("hello");
mav.addObject("message", "Hello World!");
return mav;
}
}
<bean id="sampleController" class="samples.SampleController">
<property name="cacheSeconds" value="120"/>
</bean>
请注意,手动实例化servlet并不是一个好的做法或好的设计。相反,这更像是让Spring控制器连接并连接起来的垫脚石,这样您就可以通过将业务逻辑移动到服务层来系统地重构代码并消除遗留的servlet。
这种方法符合您的计划,系统地迁移并了解有关Spring的更多信息,解决一些技术债务,同时仍然致力于与您的网站相关的业务目标。