JavaEE forward()和sendRedirect() 总结

来源:互联网 发布:app和服务器数据同步 编辑:程序博客网 时间:2024/05/17 05:17



        页面之间的跳转可以通过两种方式实现:forward()和sendRedirect()
  forward:可以在JSP页面中使用,可以在Servlet中实现。
  在JSP页面中使用<jsp:forward page="目标文件"/>,例如要跳转到userlist.jsp,可以写:
  <jsp:forward page="userlist.jsp"/>
  在Servlet中使用RequestDispatcher的forward方法,要跳转到userlist.jsp,可以写:
  RequestDispatcher rd = request.getRequestDispatcher("userlist.jsp");
  rd.forward(request,response);
  sendRedirect方式使用response对象的方法sendRedirect方法完成,代码如下:
  response.sendRedirect("userist.jsp");
  两者的区别如下:
  1、请求次数不同,这是最本质的区别。在forward方式下,在执行当前JSP对象或者Servlet对象的过程中去调用目标文件对应的对象,相当于方法调用,把request和response对象作为参数传递到目标文件对应的对象,当前文件和目标文件的执行是在用户发送的一次请求中完成的。在 redirect方式下,用于首先请求了当前文件,当前文件把目标文件的地址返回给了客户端,客户端再次发送请求,请求目标文件,实际上是发送了两次请求。
  2、传值方式不同。在forward方式下,当前文件和目标文件属于同一次请求,共享request对象,所以可以使用request对象传值。在 redirect方式下,当前文件和目标文件属于不同的请求,每次请求会单独创建request和response对象,这样就不能使用request对象来传值。在MVC模式下,通常在控制器中调用模型得到数据,然后保存到request中,然后forward到目标文件,目标文件从request中获取需要的信息。如果使用sendRedirect方式在控制器和视图之间传递信息,需要使用在目标文件之后加上“?名字=值”的方式传递。
  3、客户端在地址栏中看到的地址不一样,对于forward,在地址栏中看到的是第1个文件的名字,对于sendRedirect,在地址栏中看到的是第2个文件的地址。有时候会影响目标文件中的相对路径,例如当前文件是aa文件夹中的a.jsp,目标文件是bb文件夹中的b.jsp,在b.jsp 中要访问一个图片,使用相对路径,直接写face.jpg,这个文件与b.jsp放在一起。如果采用forward方式,地址栏中是a.jsp,这样系统会在aa文件夹中找face.jpg,这时候就会出错。




  sendRedirect和forward原理及区别总结
  一、原理.
  1、 Forward
  该图的交互过程如下:
  ① 浏览器访问Servlet1。
  ② Servlet1想让Servlet2对客户端的请求进行响应,于是调用forward()方法,将请求转发给Servlet2进行处理。
  ③ Servlet2对请求做出响应。
  交互过程可以看出,调用forward()方法,对浏览器来说是透明的,浏览器并不知道为其服务的Servlet已经换成Servlet2了,它只知道发出了一个请求,获得了一个响应。显示的URL始终是原始请求的URL。
  sendRedirect()方法和forward()方法还有一个区别,那就是sendRedirect()方法不但可以在位于同一主机上的不同Web应用程序之间进行重定向,而且可以将客户端重定向到其他服务器上的Web应用程序资源。
  这种方式是在服务器端作的重定向。服务器往client发送数据的过程是这样的:服务器在向客户端发送数据之前,是先将数据输出到缓冲区,然后将缓冲区中数据发送给client端。什么时候将缓冲区里的数据发送给client端呢?
  (1)当对来自client的request处理完,并把所有数据输出到缓冲区
  (2)当缓冲区满
  (3)在程序中调用缓冲区的输出方法out.flush()或response.flushbuffer(),web container才将缓冲区中的数据发送给client。
  这种重定向方式是利用服务器端的缓冲区机制,在把缓冲区的数据发送到客户端之前,原来的数据不发送,将执行转向重定向页面,发送重定向页面的数据,重定向调用页的数据将被清除。如果在<JSP:FORWORD>之前有很多输出,前面的输出已使缓冲区满,将自动输出到客户端,那么这种重定向方式将不起作用,这一点应该特别注意。
  public void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException
  {
  response.setContentType("text/html; charset=UTF-8");
  ServletContext sc = getServletContext();
  RequestDispatcher rd = null;
  rd = sc.getRequestDispatcher("/index.jsp");
  rd.forward(request, response);
  }
  2、 sendRedirect
  交互过程可以看出,调用sendRedirect()方法,实际上是告诉浏览器Servlet2所在的位置,让浏览器重新访问Servlet2。调用sendRedirect()方法,会在响应中设置Location响应报头。要注意的是,这个过程对于用户来说是透明的,浏览器会自动完成新的访问。浏览器的地址栏中,可以看到,显示的URL是重定向之后的URL。
  该图的交互过程如下:
  ① 浏览器访问Servlet1。
  ② Servlet1想让Servlet2为客户端服务。
  ③ Servlet1调用sendRedirect()方法,将客户端的请求重定向到Servlet2。
  ④ 浏览器访问Servlet2。
  ⑤ Servlet2对客户端的请求做出响应。
  这种方式是在客户端作的重定向处理。该方法通过修改HTTP协议的HEADER部分,对浏览器下达重定向指令的,让浏览器对在location中指定的URL提出请求,使浏览器显示重定向网页的内容。该方法可以接受绝对的或相对的URLs。如果传递到该方法的参数是一个相对的URL,那么Web container在将它发送到客户端前会把它转换成一个绝对的URL。
  public void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException
  {
  response.setContentType("text/html; charset=UTF-8");
  response.sendRedirect("/index.jsp");
  }
  二、区别
  1.response.sendRedirect(url)
  1)重定向,不转发请求,地址栏的url已改变
  2)request.getAttribute(""),request.getParamter("")都获取不了
  response.sendRedirect("/studyProject/myforward/end2.jsp?name=tomzhang");//可以带参数的
  2.<jsp:forward page="" />
  1)请求转发,地址栏的url不变
  2)可以传递额外的参数
  <jsp:param name="" value=""/>
  3)request.getAttribute("");//基于请求的request共享
  request.getParamter("");//请求转发
  4)等同requestDispatcher.forward(url);
  <jsp:forward page="/myforward/end.jsp?age=man">//可以带参数
  *使用forward一定要注意url页面的链接,不要使用相对路径;url页面的当前路径其实是第1次请求时的当前路径,所以url页面的链接都应该采用绝对路径!
  三、详解
  1. 跳转方式
  http://localhost:8080/Test应用
  运用forward方法只能重定向到同一个Web应用程序中的一个资源。而sendRedirect方法可以让你重定向到任何URL。
  表单form的action= "/uu ";sendRedirect( "/uu ");表示相对于服务器根路径。如http://localhost:8080/Test应用(则提交至http://localhost:8080/uu);
  Forward代码中的 "/uu "则代表相对与WEB应用的路径。如http://localhost:8080/Test应用(则提交至http://localhost:8080/Test/uu);
  2. (运用RequestDispatcher接口的Forward)方法
  forward()无法重定向至有frame的jsp文件,可以重定向至有frame的html文件,
  同时forward()无法在后面带参数传递,比如servlet?name=frank,这样不行,可以程序内通过response.setAttribute( "name ",name)来传至下一个页面.
  重定向后浏览器地址栏URL不变.
  只有在客户端没有输出时才可以调用forward方法。如果当前页面的缓冲区(buffer)不是空的,那么你在调用forward方法前必须先清空缓冲区。
  "/ "代表相对与web应用路径
  RequestDispatcher   rd   =   request.getRequestDispatcher( "/ooo ");
  rd.forward(request,   response);提交至http://localhost:8080/Test/ooo
  RequestDispatcher   rd   =   getServletContext().getRequestDispatcher( "/ooo ");
  rd.forward(request,   response);提交至http://localhost:8080/Test/ooo
  RequestDispatcher   rd   =getServletContext().getNamedDispatcher( "TestServlet ");(TestServlet为一个 <servlet-name> )
  rd.forward(request,   response);提交至名为TestServlet的servlet
  如果在 <jsp:forward> 之前有很多输出,前面的输出已使缓冲区满,将自动输出到客户端,那么该语句将不起作用,这一点应该特别注意。
  另外要注意:它不能改变浏览器地址,刷新的话会导致重复提交
  从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
  (运用HttpServletResponse接口的sendRedirect)方法302
  是在用户的浏览器端工作,sendRedirect()可以带参数传递,比如servlet?name=frank传至下个页面,
  同时它可以重定向至不同的主机上,sendRedirect()可以重定向有frame.的jsp文件.
  假设转发代码包含于注册的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,
  sendRedirect等同于此方式
  response.setStatus(HttpServletResponse.SC_MOVED_PERMANENTLY);
  String   newLocn   =   "/newpath/jsa.jsp ";
  response.setHeader( "Location ",newLocn);
  (Meta   Refresh)方法200
  这种方法是由HTML提供的,Meta本身就是HTML标签。使用方法是: <meta   http-equiv= "refresh "   content= "5;   url=http://www.dreamdu.com/ "   />
  相应的java代码
  String   content=stayTime+ ";URL= "+URL;
  response.setHeader( "REFRESH ",content);
  3. 使用response.sendRedirect()地址栏将改变
  使用request.getRequestDispatcher().forward(request,response)地址栏中的信息保持不变
  4. request.setAttribute存的东西
  只用通过方法2跳转   才能在新页取出来
  5. redirect   会首先发一个response给浏览器,   然后浏览器收到这个response后再发一个requeset给服务器,然后服务器发新的response给浏览器。时页面收到的request是一个新从浏览器发来的。
  forward   发生在服务器内部,在浏览器完全不知情的情况下发给了浏览器另外一个页面的response。这时页面收到的request不是从浏览器直接发来了,可能己经放了数据。
  所以: request.setAttribute存的东西,只用通过方法2跳转,才能在新页取出来。





        HttpServletResponse.sendRedirect与RequestDispatcher.forward方法都可以实现获取相应URL资源。
  sendRedirect实现请求重定向,forward实现的是请求转发。
  在web服务器内部的处理机制也是不一样的。
  1. 跳转方式
  运用forward方法只能重定向到同一个Web应用程序中的一个资源。而sendRedirect方法可以让你重定向到任何URL。
  表单form的action= "/uu ";sendRedirect( "/uu ");表示相对于服务器根路径。
  如http://localhost:8080/Test应用(则提交至http://localhost:8080/uu);
  Forward代码中的 "/uu "则代表相对与WEB应用的路径。如http://localhost:8080/Test应用(则提交至http://localhost:8080/Test/uu);
  2.forward重定向后,浏览器url地址不变,sendRedirect转发后,浏览器url地址变为目的url地址。
  forward()无法重定向至有frame的jsp文件,可以重定向至有frame的html文件, 同时forward()无法在后面带参数传递,
  比如servlet?name=frank,这样不行,可以程序内通过response.setAttribute( "name ",name)来传至下一个页面.
  3. 使用forward重定向的过程,是浏览器先向目的Servlet发送一次Request请求,然后再服务器端由Servlet再将请求发送到目的url,再由服务器端Servlet返回Response到浏览器端。浏览器和服务器一次请求响应。
  使用sendRedirect转发的过程,浏览器先向目的Servlet发送一次请求,Servlet看到sendRedirect将目的url返回到浏览器,浏览器再去请求目的url,目的url再返回response到浏览器。浏览器和服务器两次请求响应。
  4. forward方法的调用者与被调用者之间共享Request和Response
  sendRedirect方法由于两次浏览器服务器请求,所以有两个Request和Response。
  如果使用request.setAttribute传递一些属性就需要用forward,如果想要跳转到别的应用的资源,就需要用sendRedirect。
  5.无论是forward方法还是sendRedirect方法调用前面都不能有PrintWriter输出到客户端。
  forward方法报错: java.lang.IllegalStateException: Cannot forward after response has been committed
  sendRedirect报错:java.lang.IllegalStateException
  at org.apache.catalina.connector.ResponseFacade.sendRedirect(ResponseFacade.java:435)

原创粉丝点击