forward 和redirect的区别

来源:互联网 发布:洛克希德马丁公司知乎 编辑:程序博客网 时间:2024/06/08 09:48
forwardredirect的区别

forward是服务器请求资源,服务器直接访问目标地址的URL,把那个URL的响应内容读取过来,然后把这些内容再发给浏览器,浏览器根本不知道服务器发送的内容是从哪儿来的,所以它的地址栏中还是原来的地址。 

redirect就是服务端根据逻辑,发送一个状态码,告诉浏览器重新去请求那个地址,一般来说浏览器会用刚才请求的所有参数重新请求,所以session,request参数都可以获取。


在Servlet程序中,有时需要调用服务器端的另外一个资源来对浏览器的请求进行响应,这可以通两种方式来实现;

其中一种就是调用 RequestDispatcher.forward 方法与 include 方法实现的请求转发;

另外一种则是调用 HttpServletResponse.sendRedirect 方法实现的请求重定向.

1: RequestDispatcher 接口(请求转发)   

RequestDispatcher 接口中定义了两个方法: 

forward 方法用于将请求转发到 RequestDispatcher 对象封装的资源
include方法将 RequestDispatcher 对象封装的资源作为当前响应内容的一部分包含进来.

1) RequestDispatcher对象的获取[通过ServletContext对象]:      ServletContext 接口中定义了两个用于获取 RequestDispatcher对象的方法      

(1) getRequestDispatcher(String param)      
(2) getNamedDispatcher(String param)

ServletContext sc = this.getServletContext();  //ServletContext对象的获取//RequestDispatcher对象的获取RequestDispatcher dis = sc.getRequestDispatcher("/servlet2");  RequestDispatcher dis = sc.getRequestDispatcher("/servlets/targetservlet");RequestDispatcher dis = sc.getRequestDispatcher("/servlets/targetservlet?userName=" + uName);

[注: 传递给该方法的路径字符串必须以"/"开头,这个"/"代表当前WEB应用程序的根目录] 
ServletContext sc = this.getServletContext();  //ServletContext对象的获取
[注: 传递给该方法的参数是在WEB应用程序部署描述符中声明过的Servlet或jsp的映射名称] 
2) RequestDispatcher对象的获取[通过ServletRequest对象]:      
在ServletRequest接口中也定义了一个 getRequestDispatcher(String param) 方法来获取 RequestDispatcher对象,它与ServletContext.getRequestDispatcher()方法的区别在于:  传递给这个方法的参数除了可以采用以"/"开头的路径字符串,还可以采用非"/"开头的相对路径.      
(1) getRequestDispatcher(String param)

3) 用forward方法实现请求转发      dis.forward(request, response); //[两个页面中的request与response是相同的,不过响应内容将被清空]   
4) 用include方法实现资源包含      dis.include(request, response); //[两个页面中的request与response是相同的]

2: HttpServletResponse 接口(请求重定向)   HttpServletResponse 接口中定义了一些用于将请求再次定向到新的资源下的方法,最主要的是   

(1) sendRedirect(String url)   该方法不仅可以重定向到当前应用程序中的其他资源,它还可以重定向到同一个站点上的其他   应用程序中的资源,甚至是使用绝对URL重定向到其他站点的资源.而RequestDispatcher.forward   方法只能在同一个WEB应用程序内资源之间转发请求.如果传递给sendRedirect 方法的相对URL不   是以"/"开头,则表示是相对于当前请求的URL;如果该相对URL是以"/"开头,则是相对于整个WEB站   点的根目录,而不是相对于当前WEB应用程序的根目录,例如,下面的语句表示重定向到当前站点中   的APTECH应用程序根目录下的index.html页面;   response.sendRedirect("/APTECH/index.html");


1.从地址栏显示来说forward是服务器请求资源,即服务器直接访问目标地址的URL,把那个URL的响应内容读取过来,然后把这些内容再发给浏览器,浏览器根本不知道服务器发送的内容从哪里来的,所以它的地址栏还是原来的地址.redirect是服务端根据逻辑,发送一个状态码,告诉浏览器重新去请求那个地址.所以地址栏显示的是新的URL.
2.从数据共享来说forward:转发页面和转发到的页面可以共享request里面的数据.redirect:不能共享数据.
3.从运用地方来说forward:一般用于用户登陆的时候,根据角色转发到相应的模块.redirect:一般用于用户注销登陆时返回主页面和跳转到其它的网站等.
4.从效率来说forward:高.redirect:低.下面罗列三种网络上的具体解释:

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

 转发过程:客户浏览器发送http请求 --> web服务器接受此请求 --> 调用内部的一个方法在容器内部完成请求处理和转发动作- --> 将目标资源发送给客户;在这里,转发的路径必须是同一个web容器下的url,其不能转向到其他的web路径上去,中间传递的是自己的容器内的request。在客户浏览器路径栏显示的仍然是其第一次访问的路径,也就是说客户是感觉不到服务器做了转发的。转发行为是浏览器只做了一次访问请求。重定向过程:客户浏览器发送http请求----》web服务器接受后发送302状态码响应及对应新的location给客户浏览器--》客户浏览器发现是302响应,则自动再发送一个新的http请求,请求url是新的location地址----》服务器根据此请求寻找资源并发送给客户。在这里location可以重定向到任意URL,既然是浏览器重新发出了请求,则就没有什么request传递的概念了。在客户浏览器路径栏显示的是其重定向的路径,客户可以观察到地址的变化的。重定向行为是浏览器做了至少两次的访问请求的。

解释二 重定向(redirect)其实是两次request,第一次,客户端request   A,服务器响应,并response回来,告诉浏览器,你应该去B。这个时候IE可以看到地址变了,而且历史的回退按钮也亮了。重定向可以访问自己web应用以外的资源。在重定向的过程中,传输的信息会被丢失。请求转发(forward)是服务器内部把对一个request/response的处理权,移交给另外一个 对于客户端而言,它只知道自己最早请求的那个A,而不知道中间的B,甚至C、D。 传输的信息不会丢失。 

解释三 假设你去办理某个执照, 重定向(redirect):你先去了A局,A局的人说:“这个事情不归我们管,去B局”,然后,你就从A退了出来,自己乘车去了B局。 转发(forward):你先去了A局,A局看了以后,知道这个事情其实应该B局来管,但是他没有把你退回来,而是让你坐一会儿,自己到后面办公室联系了B的人,让他们办好后,送了过来。

具体代码实现中:在Servlet中两种实现: 

forward方式:request.getRequestDispatcher("/somePage.jsp").forward(request, response);
redirect方式:response.sendRedirect("/somePage.jsp");

forward是服务器内部重定向,程序收到请求后重新定向到另一个程序,客户机并不知道; 

redirect则是服务器收到请求后发送一个状态头给客 户,客户将再请求一次,这里多了两次网络通信的来往。当然forward也有缺点,就是forward的页面的路径如果是相对路径就会有些问题了。



原创粉丝点击