一、调用方式
我们知道,在servlet中调用转发、重定向的语句如下:
request.getRequestDispatcher("new.jsp").forward(request, response);//转发到new.jsp
response.sendRedirect("new.jsp");//重定向到new.jsp
在jsp页面中你也会看到通过下面的方式实现转发:
<jsp:forward page="apage.jsp" />
当然也可以在jsp页面中实现重定向:
<%response.sendRedirect("new.jsp");//重定向到new.jsp%>
二、本质区别
解释一
1,请求重定向:客户端行为,response.sendRedirect(),从本质上讲等同于两次请求,前一次的请求对象不会保持,地址栏的URL地址会改变。
2,请求转发:服务器行为,request.getRequsetDispatcher().forward(requset,response);是一次请求,转发后请求对象会保存,地址栏的URL地址不会改变。(服务器内部转发,所有客户端看不到地址栏的改变)
一句话,转发是服务器行为,重定向是客户端行为。为什么这样说呢,这就要看两个动作的工作流程:
转发过程:客户浏览器发送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。 传输的信息不会丢失。
解释三
请求转发是服务端行为:用户发送一个HTTP请求>>>服务器接收到请求>>>调用内部方法在作用域(项目)中完成请求处理和转发>>>将资源相应给客户端。服务器转发的地址只能是域(项目)中的路径,不能跳转到其他域(项目)的路径上,转发后的URL地址栏不变,只能通过浏览器开发者模式查看到对应请求转发的地址,客户端只请求了一次服务端。
重定向是客户端行为:用户发送一个HTTP请求>>>服务器接收到发送302状态码相应和重定向的地址>>>客户端接受到状态码和定向的URL地址>>>客户端重新发送新的HTTP请求(请求的路径是服务端相应给客户端的URL路径)>>>服务端接受到客户端再次请求的数据并相应给客户端数据。所以浏览器的URL地址栏是会编发的,而且在两次请求中不能共享数据。
301,302都是HTTP状态的编码,都带便这URL发生了变化。301表示永久性跳转,302表示暂时性跳转。如果不是暂时性转译推荐使用301永久性跳转,301永久性跳转对搜索引擎优化更好。