【问题标题】:Using sendRedirect inside a servlet forwarded with request dispatcher在使用请求调度程序转发的 servlet 中使用 sendRedirect
【发布时间】:2013-05-19 14:33:05
【问题描述】:

我有一个过滤器,它实现了加载 servlet 和 JSP 的自定义约定。在该约定中,我使用以下代码来包含 servlet:

servletContext
    .getRequestDispatcher( uriDispatcherLocator.getServletLocation( uri ) )
    .include( request, response );

以及以下代码以包含 JSP(在同一个过滤器中):

servletContext
    .getRequestDispatcher( "/index.jsp" )
    .include( request, response );

一切正常,servlet 执行,然后它包含 JSP 和一些不相关的自定义规则。 如您所见,目前我在请求调度程序中包含一个 servlet我无法向客户端发送 http 标头响应。 问题是我希望 servlet 能够完全控制响应,就好像它是从过滤器内部调用的一样(因为过滤器只会根据项目文件系统中它们各自的 Class/JSP 位置动态映射 servlet )。 我可以使用.forward() 代替.include(),但如果这样做,我将无法在servlet 执行后包含JSP。

那么,当通过 RequestDispatcher 接口通过过滤器包含时,我如何允许 servlet 执行下面的代码?

response.sendRedirect( "/somePath" );

没有 Javascript hack,我愿意从服务器发送正确的 HTTP 响应以使浏览器正常运行。

--

编辑: 换句话说:

我想使用 RequestDispatcher 更改从 INSIDE 包含的 servlet 发送到客户端的标头,但文档指出:

The included servlet cannot change the response status code or set headers; any attempt to make a change is ignored.

【问题讨论】:

  • 如果要发送重定向,则无法呈现 jsps。我可能误解了您要实现的目标。
  • 您可以在 servlet 中抛出异常来指示过滤器进行重定向。不是最优雅的解决方案,但至少重定向的决定可以保留在 servlet 中......
  • @SotiriosDelimanolis 我将 servlet 内容包含在响应中。我想包含 servlet,能够操作标头并包含带有过滤器的 jsp。
  • @SirRotN 我不想只是重定向。有时应该需要以某种方式操作标头,因此我不能为每种行为创建一个异常类型(尽管这有点道理)

标签: java servlets servlet-filters


【解决方案1】:

您的过滤器包含您的 servlet

servletContext
    .getRequestDispatcher( uriDispatcherLocator.getServletLocation( uri ) )
    .include( request, response );


您的 Servlet 表明它想要重定向

request.setAttribute ("sendRedirect", "/some/path");


或者,希望添加一个或多个响应标题

Map<String, String> respHeaders = new HashMap<String, String>();
respHeaders.put("Expires", "0");
respHeaders.put("Cache-Control", "no-cache");
request.setAttribute("respHeaders", respHeaders);


您的过滤器会检查这些特殊要求

Map<String, String> respHeaders =
                   (Map<String, String>) request.getAttribute("respHeaders");
for (String key : respHeaders.keySet()) {
    response.setHeader(key, respHeaders.get(key)); // set response headers
}

String sendRedirect;
if ((sendRedirect = (String) request.getAttribute("sendRedirect")) != null) {
    response.sendRedirect(sendRedirect); // redirect the client
}


编辑:当从包含外部调用时,也许您的一些 servlet 已经在驱动流程并设置响应标头。但是,没有办法按原样重用它们。您不能简单地包含它们并期望相同的行为,因为 RequestDispatcher#include() 的目标是仅提供服务器端包含 (SSI)。

因此,我们在 API 中找不到任何重载方法(或任何可以修改此行为的设置器)。如果您想包含此类 servlet 并保留它们的行为(如重定向),则必须向它们传递它们在 include 上下文中运行的提示,因此应提交它们的响应请求。

request.setAttribute ("includeContext", true);

【讨论】:

  • 这是一个简单且非常好的解决方案。不幸的是,它需要超出规范的额外控制。我可能正在寻找一种解决方案,该解决方案使用一些允许此类行为的 API,而无需创建自定义实现(如果存在的话)。
  • 该解决方案实际上完全遵守规范。该规范将包含的 Web 组件视为“来宾”,即它不能引导流程,并且任何此类尝试都将被正确忽略(而不是可能允许包含您可能拥有的任何 servlet 的例外)。该规范声明只有主机 Web 组件(执行包含的组件)才能完全控制流程以及将哪些响应标头发送到客户端。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多