目录
1.实现
2.具体区别
1.有关实现
请求转发与重定向分别对应forward 和 redirect两个关键字,接下来我们在Java中尝试去实现一下。
1.1 请求转发
我们一般使用两种方式实现,具体代码见下:
@RequestMapping("/fw")
public String myForward(){
return "forward:login.html";//此处forward可以省略
}
@RequestMapping("/fw2")
public void myForward2(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
request.getRequestDispatcher("/login.html").forward(request,response);
}
1.2 请求重定向
常用的实现方法同样有两种,具体代码见下:
@RequestMapping("/rd")
public String myRedirect(){
return "redirect:login.html";//此处redirect可以省略
}
@RequestMapping("/rd2")
public void myRedirect2(HttpServletResponse response) throws IOException {
response.sendRedirect("login.html");
}
2.具体区别
虽然这两种方法从表面上看都是实现了页面跳转,但实际上他们大有不同,主要体现在以下五个方面:
1.定义不同
2.请求方不同
3.数据共享不同
4.最终 URL 地址不同
5.代码实现不同
2.1 定义不同
请求转发(Forward):发生在服务端程序内部,当服务器端收到一个客户端的请求之后,会先将请求,转发给目标地址,再将目标地址返回的结果转发给客户端。 而客户端对于这一切毫无感知的,这就好比,张三(客户端)找李四(服务器端)借钱,而李四没钱,于是李四又去王五那借钱,并把钱借给了张三,整个过程中张三只借了一次款,剩下的事情都是李四完成的,这就是请求转发。
请求重定向(Redirect):请求重定向指的是服务器端接收到客户端的请求之后,会给客户端返回了一个临时响应头,这个临时响应头中记录了,客户端需要再次发送请求(重定向)的 URL 地址,客户端再收到了地址之后,会将请求发送到新的地址上,这就是请求重定向。这就好像张三(客户端)找李四(服务器端)借钱,李四没钱,于是李四就告诉张三,“我没钱,你去王五那借“,于是张三又去王五家借到了钱,这就是请求重定向。
2.2 请求方不同
从定义我们可以得知请求转发应该只发送了一个HTTP请求,而请求重定向应该发送了两个HTTP请求。而这一点我们可以通过fiddler抓包来验证。
我们先看到请求重定向:
可以看到确实只发送了一个HTTP请求,并且是直接将目标页面返回了
接下来看到请求转发:
可以看到确实如定义所说,请求转发是先返回目标页面的地址,然后让我们再去访问目标页面最后才得到信息
从上面请求转发和请求重定向的定义,我们可以看出:请求转发是服务器端的行为,服务器端代替客户端发送请求,并将结果返回给客户端;而请求重定向是客户端的行为,它们的交互流程,如下图所示:
2.3 数据共享不同
请求转发是服务器端实现的,整个执行流程中,客户端(浏览器端)只需要发送一次请求,因此整个交互过程中使用的都是同一个 Request 请求对象和一个 Response 响应对象,所以整个请求过程中,请求和返回的数据是共享的;而请求重定向客户端发送两次完全不同的请求,所以两次请求中的数据是不同的。
2.4 最终 URL 地址不同
请求转发是服务器端代为请求,再将结果返回给客户端的,所以整个请求的过程中 URL 地址是不变的;而请求重定向是服务器端告诉客户端让客户端去访问另一个URL,所以浏览器会重新再发送一次请求,因此客户端最终显示的 URL 也为最终跳转的地址,而非刚开始请求的地址,所以 URL 地址发生了改变。
请求重定向前后URL对比:
可以看到URL并未发生改变
请求重定向前后URL对比:
URL被重新定向到了我们想要访问的目标网址,这个过程由客户端完成,所以URL发生了改变。
2.5 代码实现不同
关于这点非常显而易见,在前面讲解代码实现时大家也可以自行去对比。
2.6 请求转发 forward 可能导致的问题
请求转发如果资源和转发的⻚⾯不在⼀个⽬录下,会导致外部资源不可访问。
关于这点我们很好理解,因为请求转发前后URL并未改变,而网页中的一些资源路径是通过相对路径设置的,假如请求转发的页面是一级目录,而目标页面是二级目录,那么此时资源文件在转发页面的一级目录通过相对路径自然无法获取到所需的资源。