forward和redirect的区别

1.从地址栏显示来讲
forward是服务器请求资源,服务器直接访问目标地址的URL,把那个URL的响应内容读取过来,而后把这些内容再发给浏览器.浏览器根本不知道服务器发送的内容从哪里来的,因此它的地址栏仍是原来的地址.
redirect是服务端根据逻辑,发送一个状态码,告诉浏览器从新去请求那个地址.因此地址栏显示的是新的URL.
2.从数据共享来讲
forward:转发页面和转发到的页面能够共享request里面的数据.
redirect:不能共享数据.
3.从运用地方来讲
forward:通常用于用户登录的时候,根据角色转发到相应的模块.
redirect:通常用于用户注销登录时返回主页面和跳转到其它的网站等.
4.从效率来讲
forward:高.
redirect:低.
本质区别java

解释一  web

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

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

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

解释二
重定向,实际上是两次request,
第一次,客户端request A,服务器响应,并response回来,告诉浏览器,你应该去B。这个时候IE能够看到地址变了,并且历史的回退按钮也亮了。重定向能够访问本身web应用之外的资源。在重定向的过程当中,传输的信息会被丢失。post


请求转发是服务器内部把对一个request/response的处理权,移交给另一个
对于客户端而言,它只知道本身最先请求的那个A,而不知道中间的B,甚至C、D。 传输的信息不会丢失。网站


解释三this

假设你去办理某个执照,url

重定向:你先去了A局,A局的人说:“这个事情不归咱们管,去B局”,而后,你就从A退了出来,本身乘车去了B局。spa

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

3、请求重定向与请求转发的比较

尽管HttpServletResponse.sendRedirect方法和RequestDispatcher.forward方法均可以让浏览器获 得另一个URL所指向的资源,但二者的内部运行机制有着很大的区别。下面是HttpServletResponse.sendRedirect方法实现 的请求重定向与RequestDispatcher.forward方法实现的请求转发的总结比较:

(1)RequestDispatcher.forward方法只能将请求转发给同一个WEB应用中的组件;而 HttpServletResponse.sendRedirect 方法不只能够重定向到当前应用程序中的其余资源,还能够重定向到同一个站点上的其余应用程序中的资源,甚至是使用绝对URL重定向到其余站点的资源。若是 传递给HttpServletResponse.sendRedirect 方法的相对URL以“/”开头,它是相对于整个WEB站点的根目录;若是建立RequestDispatcher对象时指定的相对URL以“/”开头,它 是相对于当前WEB应用程序的根目录。

(2)调用HttpServletResponse.sendRedirect方法重定向的访问过程结束后,浏览器地址栏中显示的URL会发生改变,由初 始的URL地址变成重定向的目标URL;而调用RequestDispatcher.forward 方法的请求转发过程结束后,浏览器地址栏保持初始的URL地址不变。

(3)HttpServletResponse.sendRedirect方法对浏览器的请求直接做出响应,响应的结果就是告诉浏览器去从新发出对另一 个URL的 访问请求,这个过程比如有个绰号叫“浏览器”的人写信找张三借钱,张三回信说没有钱,让“浏览器”去找李四借,并将李四如今的通讯地址告诉给了“浏览 器”。因而,“浏览器”又按张三提供通讯地址给李四写信借钱,李四收到信后就把钱汇给了“浏览器”。可见,“浏览器”一共发出了两封信和收到了两次回复, “浏览器”也知道他借到的钱出自李四之手。RequestDispatcher.forward方 法在服务器端内部将请求转发给另一个资源,浏览器只知道发出了请求并获得了响应结果,并不知道在服务器程序内部发生了转发行为。这个过程比如绰号叫“浏 览器”的人写信找张三借钱,张三没有钱,因而张三找李四借了一些钱,甚至还能够加上本身的一些钱,而后再将这些钱汇给了“浏览器”。可见,“浏览器”只发 出了一封信和收到了一次回复,他只知道从张三那里借到了钱,并不知道有一部分钱出自李四之手。

(4)RequestDispatcher.forward方法的调用者与被调用者之间共享相同的request对象和response对象,它们属于同 一个访问请求和响应过程;而HttpServletResponse.sendRedirect方法调用者与被调用者使用各自的request对象和 response对象,它们属于两个独立的访问请求和响应过程。对于同一个WEB应用程序的内部资源之间的跳转,特别是跳转以前要对请求进行一些前期预处 理,并要使用HttpServletRequest.setAttribute方法传递预处理结果,那就应该使用 RequestDispatcher.forward方法。不一样WEB应用程序之间的重定向,特别是要重定向到另一个WEB站点上的资源的状况,都应该 使用HttpServletResponse.sendRedirect方法。

(5)不管是RequestDispatcher.forward方法,仍是HttpServletResponse.sendRedirect方法,在调用它们以前,都不能有内容已经被实际输出到了客户端。若是缓冲区中已经有了一些内容,这些内容将被从缓冲区中清除。

zend framework 中的应用

控制这种分发过程的动做控制器方法是_forward()
;在任意的pre/postDispatch()
或者动做中调用该方法,并传入动做、控制器、模块、以及可选的附加参数,就能够进入新的动做。

三种请求转发得方式

public function fooAction() {

// forward to another action in the current controller and module:
$this->_forward('bar', null, null, array('baz' => 'bogus'));
}

public function barAction() {

// forward to an action in another controller:
// FooController::bazAction(),
// in the current module:
$this->_forward('baz', 'foo', null, array('baz' => 'bogus'));
}

public function bazAction() {

// forward to an action in another controller in another module,
// Foo_BarController::bazAction():
$this->_forward('baz', 'bar', 'foo', array('baz' => 'bogus'));
}

页面跳转得方法是

$this->_redirect('/login/add');
相关文章
相关标签/搜索