Servlet的几种跳转

来源:互联网 发布:男女对唱的网络歌曲 编辑:程序博客网 时间:2024/05/20 03:48

Servlet:
当然,在servlet中,一般跳转都发生在doGet, doPost等方法里面。

一、原理
1) redirect 方式
response.sendRedirect("/a.jsp");
页面的路径是相对路径。sendRedirect可以将页面跳转到任何页面,不一定局限于本web应用中,如:
response.sendRedirect("http://www.ycul.com");

跳转后浏览器地址栏变化。
这种方式要传值出去的话,只能在url中带parameter或者放在session中,无法使用request.setAttribute来传递。

这种方式是在客户端作的重定向处理。该方法通过修改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");
}



2) forward方式
RequestDispatcher dispatcher = request.getRequestDispatcher("/a.jsp");
dispatcher .forward(request, response);
页面的路径是相对路径。forward方式只能跳转到本web应用中的页面上。

跳转后浏览器地址栏不会变化。
使用这种方式跳转,传值可以使用三种方法:url中带parameter,session,request.setAttribute

这种方式是在服务器端作的重定向。服务器往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);
}

二、区别.1、forward重定向是在容器内部实现的同一个Web应用程序的重定向,所以forward方法只能重定向到同一个Web应用程序中的一个资源,重定向后浏览器地址栏URL不变,而sendRedirect方法可以重定向到任何URL, 因为这种方法是修改http头来实现的,URL没什么限制,重定向后浏览器地址栏URL改变。2、forward重定向将原始的HTTP请求对象(request)从一个servlet实例传递到另一个实例,而采用sendRedirect方式两者不是同一个application。3、基于第二点,参数的传递方式不一样。forward的form参数跟着传递,所以在第二个实例中可以取得HTTP请求的参数。sendRedirect只能通过链接传递参数,response.sendRedirect(“login.jsp?param1=a”)。4、sendRedirect能够处理相对URL,自动把它们转换成绝对URL,如果地址是相对的,没有一个‘/’,那么Web container就认为它是相对于当前的请求URI的。比如,如果为response.sendRedirect("login.jsp"),则会从当前servlet 的URL路径下找login.jsp: http://10.1.18.8:8081/dms/servlet/Servlet 重定向的URL: http://10.1.18.8:8081/dms/servlet/login.jsp,如果为response.sendRedirect("/login.jsp")则会从当前应用径下查找url:http://10.1.18.8:8081/login.jsp。而forward不能这样处理相对路径。 java 他们的区别是:response.sendRedirect是向客户浏览器发送页面重定向指令,浏览器接收后将向web服务器重新发送页面请求,所以执行完后浏览器的url显示的是跳转后的页面。跳转页面可以是一个任意的url(本服务器的和其他服务器的均可)。RequestDispatcher.forward则是直接在服务器中进行处理,将处理完后的信息发送给浏览器进行显示,所以完成后在url中显示的是跳转前的页面。在forward的时候将上一页面中传送的request和response信息一同发送给下一页面(而response.sendRedirect不能将上一页面的request和response信息发送到下一页面)。由于forward是直接在服务器中进行处理,所以forward的页面只能是本服务器的。


JSP:
1) response.sendRedirect();
和servlet的response.sendRedirect()方式一样。

此语句前不允许有out.flush(),如果有,会有异常:
java.lang.IllegalStateException: Can't sendRedirect() after data has committed to the client.
at com.caucho.server.connection.AbstractHttpResponse.sendRedirect(AbstractHttpResponse.java:558)
...
跳转后浏览器地址栏变化
如果要跳到不同主机下,跳转后,此语句后面的语句会继续执行,如同新开了线程,但是对response的操作已经无意义了;
如果要跳到相同主机下,此语句后面的语句执行完成后才会跳转;


2) response.setHeader("Location","");
此语句前不允许有out.flush(),如果有,页面不会跳转。
跳转后浏览器地址栏变化
此语句后面的语句执行完成后才会跳转

3) <jsp:forward page="" />
此语句前不允许有out.flush(),如果有,会有异常:
java.lang.IllegalStateException: forward() not allowed after buffer has committed.
at com.caucho.server.webapp.RequestDispatcherImpl.forward(RequestDispatcherImpl.java:134)
at com.caucho.server.webapp.RequestDispatcherImpl.forward(RequestDispatcherImpl.java:101)
at com.caucho.jsp.PageContextImpl.forward(PageContextImpl.java:836)
...
跳转后浏览器地址栏不变,但是只能跳到当前主机下
此语句后面的语句执行完成后才会跳转
 
 
 
 
 
 
 
 
 
 
 
 
 
-------------------------------------------------
 

1. 转向(Forward)

  转向(forward)是通过RequestDispatcher对象的forward(HttpServletRequest request, HttpServletResponse response)来实现的。示例如下:

?
RequestDispatcher dispatcher = request.getRequestDispatcher("/servlet/LifeCycleServlet");
dispatcher.forward(request, response);    

getRequestDispatcher()方法的参数必须以“/”开始,“/”表示本Web应用程序的根目录。如上例中,

表示要跳转的地址为http://localhost:8080/servlet/servlet/LifeCycleServlet。

forward是最常用的方式,在Structs等MVC框架中,都是用Servlet来处理用户请求,把结果通过request.setAttribute()放到request中,

然后forward到JSP中显示。

当执行forward方法时,不能有任何输出到达客户端,否则会抛出异常,也就是说,在forward之前,不要使用out.println()语句向客户端输出。

?
publicvoid doGet(HttpServletRequest request, HttpServletResponse response)
        throwsServletException, IOException {
    String destination = request.getParameter("destination");
      
    if("file".equals(destination)){
        RequestDispatcher d = request.getRequestDispatcher("/WEB-INF/web.xml");
        d.forward(request, response);
    }elseif("jsp".equals(destination)){
        request.setAttribute("date",new Date());  //attributes are reset between requests.
        RequestDispatcher dispatcher = request.getRequestDispatcher("/forward.jsp");
        dispatcher.forward(request, response);
    }elseif("servlet".equals(destination)){
        RequestDispatcher disp = request.getRequestDispatcher("/servlet/LifeCycleServlet");
        disp.forward(request, response);
    }else{
        response.setCharacterEncoding("UTF-8");
        response.getWriter().println("缺少参数。用法:"+request.getRequestURI()+"?destination=jsp或者file或者servlet");
    }
}


2. 重定向(Redirect)

  重定向是通过服务器端返回状态码来实现的。301,302都表示重定向,区别是301表示永久性重定向,302表示临时性重定向。通过sendRedirect(String location)就可以实现重定向,下面是例子。本例子主要实现了Servlet来实现文件下载并统计下载次数。要下载的文件以及下载次数都保存在一个Map中。主要思路是:首先加载页面表单,当用户点击下载链接时,客户端发起请求,运行doGet里的if判断,实现重定向。

  重定向和跳转的区别:跳转是在服务器端实现的,客户端浏览器并不知道该浏览动作,而使用Redict跳转时,跳转是在客户端实现的,也就是说客户端浏览器实际上请求了2次服务器。

?
publicclass RedictServlet extendsHttpServlet {
  
    Map<String,Integer> map =new HashMap<String,Integer>(); //new一个Map
  
    publicvoid init() throwsServletException {  //放在init中,加载servlet时运行此方法,把文件内容放到map中去
        map.put("/download/setup.exe",0);
        map.put("/download/application.zip",0);
        map.put("/download/01.mp3",0);
    }
  
    publicvoid doGet(HttpServletRequest request, HttpServletResponse response)
            throwsServletException, IOException {
        String filename = request.getParameter("filename");
          
        if(filename!=null){
            inthit = map.get(filename);  //取下载次数
            map.put(filename, ++hit);  //下载次数加1后保存
            response.sendRedirect(request.getContextPath()+filename);  //重定向到文件
              
        }else{
            response.setCharacterEncoding("UTF-8");
            PrintWriter out = response.getWriter();
            response.setContentType("text/html");
            out.println("<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\">");
            out.println("<HTML>");
            out.println("  <HEAD><TITLE>文件下载</TITLE></HEAD>");
            out.println("   <link rel='stylesheet' type='text/css' href='../css/style.css'>");
            out.println("  <BODY><br/>");
  
            out.println("<fieldset align=center style=width:90%><legend>文件下载</legend>");  //绘制页面表单
            out.println("<table width=100%>");
            out.println("   <tr>");
            out.println("       <td><b>文件名"+"</b></td>");
            out.println("       <td><b>下载次数</b></td>");
            out.println("       <td><b>下载</b></td>");
            out.println("   </tr>");
              
            for(Entry<String,Integer> entry: map.entrySet()){  //遍历map的方法
                out.println("<tr>");
                out.println("   <td>"+entry.getKey()+"</td>");
                out.println("   <td>"+entry.getValue()+"</td>");
                out.println("   <td><a href = '"+request.getRequestURI()+"?filename="+entry.getKey()+"'target = '_blank' onclick ='location = location;'>下载</a></td>");  //target='_blank'目标地址在无标题的新页面中打开。onclick ='location = location;'页面刷新
                out.println("</tr>");
            }
            out.println("</table>");
            out.println("   </legend>");
            out.println("  </BODY>");
            out.println("</HTML>");
            out.flush();
            out.close();
        }
          
    }
  
    publicvoid destroy() {
        super.destroy();// Just puts "destroy" string in log
        // Put your code here
        map =null;
    }
}