深入理解Servlet/JSP之“转发和重定向”

 

时间2008-8-5:   作者 李翊 

作者: 李翊,JAVA高级软件工程师就业班特聘讲师  blog:  (http://blog.sina.com.cn/avaj) 
作者介绍:北京航空航天大学软件工程硕士。原IBM中国研究中心(IBM China Lab)资深系统架构师,是目前国内最著名的企业级应用设计及实现专家之一。曾经负责过的主要项目包括:Petri网实验模型系统、多个省级考试中心信息系统、河南省省级电信计费系统、国务院办公厅信息系统、企业基础信息交换平台、国产信息交换平台集成测试系统、多个基于WBI企业EAI解决方案、ePod企业级电子商务解决方案、电子政务目录交换系统等 .

闻听“Bear”同学求职之时被问及“Servlet/JSP之间转发和重定向有何区别?”其实此问题不甚高明,但凡要谈区别的,必然二者非常相似并略有不同。“转发”和“重定向”本质相差甚远,答此问题只需言明二者分别为何物即可。

一、转发
   转发的原理,可以通过下图展示:

   浏览器的请求发送给组件1,组件1经过一些处理之后,将request和response对象“传递”给组件2,由组件2继续处理,然后输出响应(当然,也可以继续向其他组件“传递”),这个传递的过程称之为“转发”。整个过程只涉及一次浏览器和服务器之间的“请求-响应”,转发过程中的组件共享同一个请求(request)和响应(response)对象。
   转发的意义在于可以实现组件的“分工”。在基于MVC,多层结构的Web应用中,经常需要多个组件协同完成一次“请求-响应”工作,例如:用户要获取“设备列表信息”,提交请求至控制器组件(Servlet),该Servlet调用适当的JavaBean获取了“设备列表”数据然后再转发至JSP组件去显示信息。
   RequestDispatcher对象封装了转发操作。通过request的getRequestDispatcher(String path)方法获得RequestDispatcher对象,其中String类型参数path表示要转发到的地址。调用Dispacther对象的forward(request, response)方法实现转发。关于转发的具体操作,有如下几点需要注意:

1、转发只能在同一个应用的组件之间进行,不可以转发给其他应用的地址。

2、转发的地址可以用“相对地址”方式,也可以用“绝对地址”方式。但需要注意的是:用绝对地址方式时,应从应用名后(Context path)开始。例如:要转发到的地址为: http://192.168.5.100/tst/jsp/somewhere.jsp (其中tst为应用名),对应的绝对地址为:“/jsp/somewhere.jsp”。这是很好理解的,因为转发只能转到本应用内的地址,所以绝对地址是没有必要包含应用名的。

3、组件之间通过转发来分工协作,势必涉及数据的传递,可以通过request对象传递数据。request对象的setAttribute和getAttribute分别用于以“名称-对象对”的形式存取数据。例如:
       request.setAttribute("food", "roast duck");
       RequestDispatcher rd = request.getRequestDispatcher("jsp/somewhere.jsp");
      rd.forward(request, response);

       somewhere.jsp的内容:
       <h1>Hey, <%=request.getAttribute("food")%></h1>
在转发之前通过request的setAttribute(String name,Object obj)方法向request中存入了一个名为“food”的字符串对象“roast duck”。在转发到的jsp/somewhere.jsp中可以通过request的方法
getAttribute方法取出名为“food”的对象,然后显示在页面上。

4、在一个组件转发给另外一个组件之前,通过response输出的响应内容是没有意义的。例如:在下面的语句中:
      response.setContentType("text/html");
     PrintWriter out = response.getWriter();
     out.println("<h2>head</h2><hr/>");
      RequestDispatcher rd = request.getRequestDispatcher("jsp/somewhere.jsp");  
      rd.forward(request, response);
用户看到的是/jsp/somewhere.jsp的内容,而在转发之前的通过out对象输出的内容:
"<h2>head</h2><hr/>" 最终不会输出到浏览器,这是由于RequestDispatcher在转发之前清空了输出缓冲区。但如果在转发之前输出的信息超出了缓冲区,或者调用了out对象的flush()方法,此响应内容已经输出到了客户端(称之为响应信息已提交),此时如果实施转发操作会抛出运行时异常:java.lang.IllegalStateException。
          
二、重定向
重定向的含义可以由下图说明:

   浏览器向某组件发出请求信息,组件向浏览器发回一个重定向响应信息,该响应信息不包含具体的数据内容,只是在响应头信息中包含了需要重定向到的地址信息,该地址可以是任何有效的URL。浏览器收到该重定向响应后会自动的向响应信息头中所指示的地址发出请求。整个重定向的过程涉及两次“请求-响应”。具体的重定向响应格式如下所示:
HTTP/1.1 302 Moved Temporarily
Server: Apache-Coyote/1.1
Location: http://192.168.5.100:8080/tst/jsp/somewhere.jsp
Content-Length: 0
Date: Mon, 30 Jun 2008 03:52:54 GMT

注意:重定向响应中包含重定向地址的部分。
重定向可以通过response的sendRedirect(String url)方法来实现,注意String类型的参数url表示重定向到的地址,需要注意的是,如果表示重定向到本应用内的一个绝对地址时,要从应用名前开始,例如:tst应用中的某个组件要重定向到本应用内的/jsp/somewhere.jsp,则重定向的绝对地址应该是:“/tst/jsp/somewhere.jsp”这一点和转发中的绝对地址表示是不同的。

 

你可能感兴趣的:(jsp,浏览器,servlet,企业应用,电子政务)