HTTP中请求与转发的细节

11 篇文章 1 订阅
8 篇文章 0 订阅

首先我引用一篇认为能比较好的解释这两个概念的区别原作者地址
一、调用方式

我们知道,在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

二、本质区别

解释一

一句话,转发是服务器行为,重定向是客户端行为。为什么这样说呢,这就要看两个动作的工作流程:

转发过程:客户浏览器发送http请求——》web服务器接受此请求——》调用内部的一个方法在容器内部完成请求处理和转发动作——》将目标资源发送给客户;在这里,转发的路径必须是同一个web容器下的url,其不能转向到其他的web路径上去,中间传递的是自己的容器内的request。在客户浏览器路径栏显示的仍然是其第一次访问的路径,也就是说客户是感觉不到服务器做了转发的。转发行为是浏览器只做了一次访问请求。

重定向过程:客户浏览器发送http请求——》web服务器接受后发送302状态码响应及对应新的location给客户浏览器——》客户浏览器发现是302响应,则自动再发送一个新的http请求,请求url是新的location地址——》服务器根据此请求寻找资源并发送给客户。在这里location可以重定向到任意URL,既然是浏览器重新发出了请求,则就没有什么request传递的概念了。在客户浏览器路径栏显示的是其重定向的路径,客户可以观察到地址的变化的。重定向行为是浏览器做了至少两次的访问请求的。

解释二

重定向,其实是两次request

第一次,客户端request A,服务器响应,并response回来,告诉浏览器,你应该去B。这个时候IE可以看到地址变了,而且历史的回退按钮也亮了。重定向可以访问自己web应用以外的资源。在重定向的过程中,传输的信息会被丢失。

例子:

response.sendRedirect(“loginsuccess.jsp”);

请求转发是服务器内部把对一个request/response的处理权,移交给另外一个

对于客户端而言,它只知道自己最早请求的那个A,而不知道中间的B,甚至C、D。传输的信息不会丢失。

例子:

   RequestDispatcher dis=request.getRequestDispatcher(“loginsuccess.jsp”);

   Dis.forward(request,response);

解释三

假设你去办理某个执照

重定向:你先去了A局,A局的人说:“这个事情不归我们管,去B局”,然后,你就从A退了出来,自己乘车去了B局。

转发:你先去了A局,A局看了以后,知道这个事情其实应该B局来管,但是他没有把你退回来,而是让你坐一会儿,自己到后面办公室联系了B的人,让他们办好后,送了过来。

———————————————————我是分割线———————————————————————–

接下来我想解释下一些细节性的东西(关于缓冲区与头信息)

  • RequestDispatcher.forward方法只能将请求转发给方 法只能将请求转发给同一个WEB应用中的组件 ,而HttpServletResponse.sendRedirect方法还可以重定向到同一站点上的其它应用程序中的资源,甚至是使用绝对URL可以重定向到其它站点的资源。
  • 如果传递给HttpServletResponse.sendRedirect方法的相对URL以“/”开头,他是相当于服务器的根目录;如果创建RequestDispatcher对象时指定的相对URL以“/”开头,他是相当于当前web应用程序的根目录。
  • 调用HttpServletResponse.sendRedirect方法重定向的访问过程结束后,浏览器地址栏显示的URL会发生改变,由初始的URL地址变成重定向的目标URL调用RequestDispatcher.forward方法的请求转发过程结束后,浏览器地址栏保持初始的URL地址不变。
  • HttpServletResponse.sendRedirect方法对浏览器的请求之际做出响应,响应的结果就是告诉浏览器去重新发出对另外一个URL的访问请求:RequestDispatcher.forward方法在服务端内部将请求转发给另外一个资源,浏览器只知道发出了请求并得到了响应结果,并不知道在服务器程序内部发生了转发行为。
  • RequestDispatcher.forward方法的调用者与被调用者之间共享相同的request对象和response对象,它们属于同一个访问请求和响应过程;而 HttpServletResponse.sendRedirect方法调用者与被调用者使用各自的request对象和response对象,它们属于两个独立的访问请求和响应过程。
  • 如果在RequestDispatcher.forward方法方法之前,在Servlet程序中写入的部分内容已经被真正地传送到了客户端,forward方法将抛出IllgalStateException异常。
  • 如果在调用forward方法之前向Servlet引擎的缓冲区(response)中写入了内容,只要写到缓冲区的内容还没有真正输出到客户端,forward方法就可以正常执行,原来写入到缓冲区中的内容将被resetBuffer(清空缓冲区),但是,已写入到HttpServletResponse对象中的响应头字段信息保持有效。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值