Servlet跳转方式sendReDirect()和forward()

本文详细介绍了在Web应用服务中,SERVLET之间的跳转方法:RequestDispatcher.forward() 和 ServletResponse.sendReDirect()。对比了两者在跨应用访问能力、客户端显示、数据传输等方面的差异,并深入解析了forward的工作原理。
摘要由CSDN通过智能技术生成

在web应用服务中,经常会面对不同SERVLET之间的跳转,目前我们可以通过以下两种方式实现:

1.RequestDispatcher.forward()

2.ServletResponse.sendReDirect()

 

两者的区别:

1.redirect 方式可以跨应用访问,forward 只能在同一个应用中跳转。

2.forward 客户端访问服务器,服务器找到要跳转的相应URL并执行返回给客户端。客户端的地址栏显示的是原始URL

   redirect 客户端访问服务器,服务器找到要跳转的URL并将URL发给客户端,再由客户端对新的URL重新发送请求。客

   户端的地址栏显示的是第二次请求的URL

3.数据的传输:在request中的数据,forward可以访问,redirect不行,因为redirect已经是一个新的请求。

 

forward跳转的理解:

1.获得跳转

getRequestDispatcher方法以String参数描述ServletContext作用域的路径。该路径必须相对于ServletContext的根,并且以“/”开头。该方法使用路径(path)来查找servlet,用RequestDispatcher对象包装,然后返回结果对象。如果没有基于指定路径的servlet,提供一个RequestDispatcher返回路径值。

 

getNamedDispatcher方法以String参数表示ServletContext已知的servlet名称。如果找到servlet,用RequestDispatcher对象将它包装并返回。如果没有和指定名称关联的servlet,该方法必须返回null。

 

为了能够使用当前请求的相对路径(不是ServletContext根的相对路径)来获得RequestDispatcher对象,ServletRequest接口提供以下方法:

getRequestDispatcher

该方法的行为和ServletContext的同名方法类似。servlet容器使用request对象中的信息将指定的相对路径转换成完整路径。比如,以“/”为根的上下文,请求/garden/tools.html,通过ServletRequest.getRequestDispatcher("header.html") 获得的requestDispatcher实际上等同于调用ServletContext.getRequestDispatcher("/garden/header.html")。

 

request.getRequestDispatcher("/uu").forward(request,response)

this.getServletContext().getRequestDispatcher("/uu").forward(request,response)

this.getServletContext().getNamedDispatche("ServletName").forward(request, response)

 

1.1 RequestDispatcher路径的query字符串

ServletContext和ServletRequest使用path参数创建RequestDispatcher对象的方法,允许在path后添加query字符串。比如,开发人员可以通过下列代码获得RequestDispatcher:

String path = “/raisons.jsp?orderno=5”;

RequestDispatcher rd = context.getRequestDispatcher(path);

rd.include(request, response);

query字符串中指定的参数用来创建RequestDispatcher,优先于传递给包含servlet的同名的其他参数。该参数和RequestDispatcher关联,只在include或forward调用中有效。

 

2 使用RequestDispatcher

要使用requestDispatcher,servlet可以调用RequestDispatcher接口的include或forward方法。这些方法的参数可以是通过Servlet接口的service方法传递的request和response参数,也可以是规范2.3中引入的request和response包装类的子类的实例。后者,包装实例必须包装容器传递给service方法的request和response对象。

 

容器供应商必须确保向目标servlet转发请求和原请求处于同一个VM,同一个线程。

 

3 include方法

RequestDispatcher接口的include可以随时调用。include方法的目标servlet可以完整访问request对象,但是使用response对象上有一些限制:

 

它只能向response对象的ServletOutputStream或者Writer写入信息,提交响应(将内容输出到响应缓冲尾部),或者显式调用ServletResponse接口的flushBuffer方法。它不能设置header,或者调用任何影响响应header的方法。任何那样的尝试必须都被忽略。

 

3.1 包含的请求参数

除了使用getNamedDispatcher方法获得的servlet之外,include的servlet可以访问调用它的路径。设置以下请求属性:

 

javax.servlet.include.request_uri

javax.servlet.include.context_path

javax.servlet.include.servlet_path

javax.servlet.include.path_info

javax.servlet.include.query_string

 

include的servlet的这些属性通过request对象的getAttribute方法访问。

 

如果include的servlet通过getNamedDispatcher方法获得,那么这些属性没有设置。

 

4 forward方法

RequestDispatcher接口的forward方法只有在输出还未提交给客户端时,才可以由调用servlet调用。如果输出数据存在于还未提交的响应缓冲中,内容必须在目标servlet的service方法调用之前清空 (否则out.println()输出的都在缓存中,无效)。如果响应已被提交,必须抛出IllegalStateException异常。暴露给目标servlet的request对象的path元素必须映射用来获得RequestDispatcher的路径。

唯一例外是如果RequestDispatcher是通过getNamedDispatcher方法获得的,这种情况下,request对象的path元素必须必须映射最初的请求。

 

在RequestDispatcher接口的forward方法返回之前,servlet容器必须发送,提交并关闭响应内容。

 

public void doPost(HttpServletRequest request,HttpServletResponse response ) throws IOException,ServletException{
String page =  request.getParameter("page");//在forward时被URL中的page=1覆盖了
		response.setCharacterEncoding("UTF-8");
		//clear cache.
		response.setHeader("Pragma","No-cache");          
		response.setHeader("Cache-Control","no-cache");   
		response.setHeader("Cache-Control", "no-store");   
		response.setDateHeader("Expires",0);
		
		PrintWriter out = response.getWriter();
		out.println("<html>");
		out.println("<head><title>china</title></hea>");
		out.println("<body>");
		out.print("==");
		out.println("<br/>");//jsp页面换行
		out.println("</body>");
		out.println("</html>");
		
		//如果out.flush或者out.close().表示响应已经提交,抛出异常IllegalStateException异常。requestDispather.forward():不会执行.
		//out.flush();
		//out.close();		
		//跳转方式3种.
	//	request.getRequestDispatcher("/productInfo.do?page=1").forward(request,response);
}

 

 4.1 query字符串

请求转发机制负责在转发(forward)或包含(include)请求时传递query字符串。

 

5 错误处理

如果requesDispatcher的目标的servlet抛出ServletException或IOException类型运行时异常或checked异常,应当将它传递给调用servlet。所有其它异常应当包装成ServletException异常,并且异常的根在传递之前设置为原始异常。

 

 

相对路径的理解:

forward中"/"相对于web应用.
http://localhost:8080/Test/gw/page.jsp中转发
  <jsp:forward page="OtherPage.jsp"/>在JSP页面被解析后转换成pageContext.forward("OtherPage.jsp");
  "/OtherPage.jsp"提交到http://localhost:8080/Test/OtherPage.jsp
  "OtherPage.jsp"提交到http://localhost:8080/Test/gw/OtherPage.jsp
  
  <form action="/ddd"> 和 response.sendRedirect("/ddd"); 相对于服务器根路径.
  假设转发代码包含于注册的servlet-url为/ggg/tt;jsp为/ggg/tt.jsp:
  绝对路径:response.sendRedirect("http://www.brainysoftware.com")发送http://www.brainysoftware.com
  根路径:response.sendRedirect("/ooo")发送至http://localhost:8080/ooo
  相对路径:response.sendRedirect("ooo")发送至http://localhost:8080/Test/ggg/ooo,

 

 

参考文献:http://blog.csdn.net/wocongdongfanglai/archive/2007/02/05/1502615.aspx

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值