请求转发和请求重定向是WEB应用中经常用到的技术,因为其产生的结果相似,所以经常搞混,但是其实现机制和作用是完全不同的。
一.请求转发
请求转发通俗上来讲就是服务器中我一个Servlet处理之后,我可以选择交给其他的servlet再进行处理,说到请求转发我们不能不提请求派发器,就是RequestDispatcher,它能将当前请求转发到其他的Servlet组件中。
forward方法和include方法
这个RequestDispatcher类有两个方法一个叫,forward(HttpServletRequest,HttpServletResponse),另一个叫include(HttpServletRequest,HttpServletResponse),那么我们在哪获取这个请求派发器呐,答案就是在HttpRequestServlet中已经帮我们封装好了,我们只需要调用下面的函数,
request.getDispatherServlet()
这个函数需要一个参数,参数就是我们需要调用的Servlet的地址,然后我们选择是用forward还是include方法,我们这个例子体验下,项目结构是这样的
然后,两个servlet代码分别是这样的
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet(name = "MyServlet",value = "/ser")
public class MyServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.setContentType("text/html");
resp.getWriter().print("servlet1"+"<br/>");
req.getRequestDispatcher("ser2").forward(req,resp);
System.out.println("servlet1");
}
}
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet(name = "MyServlet0",value = "/ser2")
public class MyServlet0 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request,response);
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.getWriter().print("servlet2");
}
}
然后我们现在启动浏览器可以输出为,
并且控制台没有输出,但是请求成功从第一个servlet转发到了第二个servlet,
现在我们改一下第一个servlet代码,改成include方式
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet(name = "MyServlet",value = "/ser")
public class MyServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.setContentType("text/html");
resp.getWriter().print("servlet1"+"<br/>");
req.getRequestDispatcher("ser2").include(req,resp);
System.out.println("servlet1");
}
}
启动浏览器,可以看到浏览器输出
并且控制台也输出了
servlet1
所以forward方式会将请求转发给servlet并且本身的后序代码将不再执行,include方式会将转发到的servlet的处理置于自身的处理中,本身之后的代码将会继续执行
二.请求重定向
请求重定向是将需要重新定向信息告诉本地浏览器,然后由浏览器完成再次转发工作,由其调用函数就可以看出是客户端行为
response.sendRedirect("请求地址")
我们再次改写下,
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet(name = "MyServlet",value = "/ser")
public class MyServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.getWriter().print("servlet1");
resp.sendRedirect("ser2");
}
}
在请求前把浏览器的检查用F12打开,然后再次请求http://localhost:8080/blog/ser
可以看到输出为
可以看到浏览器地址已经改变了,并且在下面可以看到请求的地址为
http://localhost:8080/blog/ser2
我们并没有自己请求,是服务器发给浏览器,然后浏览器给我们做的
三.请求转发和请求重定向的区别
一、调用方式
我们知道,在servlet中调用转发、重定向的语句如下:
request.getRequestDispatcher(“new.jsp”).forward(request, response);//转发到new.jsp
response.sendRedirect(“new.jsp”);//重定向到new.jsp
在jsp页面中你也会看到通过下面的方式实现转发:
当然也可以在jsp页面中实现重定向:
<%response.sendRedirect(“new.jsp”);//重定向到new.jsp%>
二、本质区别
解释一
一句话,转发是服务器行为,重定向是客户端行为。为什么这样说呢,这就要看两个动作的工作流程:
转发过程:客户浏览器发送http请求—-》web服务器接受此请求–》调用内部的一个方法在容器内部完成请求处理和转发动作—-》将目标资源发送给客户;在这里,转发的路径必须是同一个web容器下的url,其不能转向到其他的web路径上去,中间传递的是自己的容器内的request。在客户浏览器路径栏显示的仍然是其第一次访问的路径,也就是说客户是感觉不到服务器做了转发的。转发行为是浏览器只做了一次访问请求。
重定向过程:客户浏览器发送http请求—-》web服务器接受后发送302状态码响应及对应新的location给客户浏览器–》客户浏览器发现是302响应,则自动再发送一个新的http请求,请求url是新的location地址—-》服务器根据此请求寻找资源并发送给客户。在这里location可以重定向到任意URL,既然是浏览器重新发出了请求,则就没有什么request传递的概念了。在客户浏览器路径栏显示的是其重定向的路径,客户可以观察到地址的变化的。重定向行为是浏览器做了至少两次的访问请求的。
解释二
重定向,其实是两次request,
第一次,客户端request A,服务器响应,并response回来,告诉浏览器,你应该去B。这个时候IE可以看到地址变了,而且历史的回退按钮也亮了。重定向可以访问自己web应用以外的资源。在重定向的过程中,传输的信息会被丢失。
例子:
请求转发是服务器内部把对一个request/response的处理权,移交给另外一个
对于客户端而言,它只知道自己最早请求的那个A,而不知道中间的B,甚至C、D。 传输的信息不会丢失。
例子:
解释三
假设你去办理某个执照,
重定向:你先去了A局,A局的人说:“这个事情不归我们管,去B局”,然后,你就从A退了出来,自己乘车去了B局。
转发:你先去了A局,A局看了以后,知道这个事情其实应该B局来管,但是他没有把你退回来,而是让你坐一会儿,自己到后面办公室联系了B的人,让他们办好后,送了过来。