该用户从未签到
|
重定向和转发有一个重要的不同:. ^3 c p$ t+ m0 a' D( Y1 ~
6 a% X' b' x2 z. x5 T' [6 p7 u6 x! R- P& P2 [9 T7 X r2 a0 C9 @
转发-----JSP容器将使用一个内部的方法来调用目标页面,新的页面继续处理同一个请求, 而浏览器将不会知道这个过程。
9 v3 h7 D; g. q1 B! E& d
B0 W0 o, J$ T. `# f7 y* B 重定向------是第一个页面通知浏览器发送一个新的页面请求。此时,浏览器中所显示的URL会变成新页面的URL, 而转发时 ,该URL会保持不变。 c2 @& [3 `3 J2 h. Y8 k
& J- [% P: I$ E7 p. u
重定向的速度比转发慢,因为浏览器还得发出一个新的请求。同时,由于重定向方式产生了一个新的请求,所以经过一次重定向后,request内的对象将无法使用。
2 k6 ]/ a& f% E/ t3 y. ]( p# ~3 _6 A" k4 K. w
7 E) U' y0 h$ y" O/ {/ q
5 J. G' f+ U( f8 h5 I6 a 怎么选择是重定向还是转发呢?
* s" r$ h2 G# ?# H1 W# H% ]8 z( e/ |3 J5 d. a# m; u
通常情况下转发更快,而且能保持request内的对象,所以他是第一选择。( ^9 M4 s: \" L& x) g, L$ |
1 U2 i9 w4 S- I5 d/ h: C
但是由于在转发之后,浏览器中URL仍然指向开始页面,此时如果重载当前页面,开始页面将会被重新调用。
; x* a2 i+ }! w `8 x* O( ^, g9 Q7 ^4 k# b' h" [
如果你不想看到这样的情况,则选择重定向
: o0 D' B6 @- t4 N0 ^0 ~, T/ u6 a1 b; \4 L2 X0 F
转发和重定向的区别
; r" {; ]# F3 a. d6 _# d3 {; P+ n# H! r6 ?: a0 N8 y/ t' q
转发方式/ s) A, f+ q, h
+ G0 p$ n9 b7 b: Z$ k
:request.getRequestDispatcher().forward(); - C( F4 U# c8 Y% t0 B, j
. [; k7 d) D1 A8 g7 Z# _
重定向方式
8 t2 O+ E& l8 s0 |- a7 Z) H* s e' B- J1 T; F# n# F
:response.sendRedirect(); ! @$ A' ~0 A* \0 [, ?$ T+ l$ o/ D
2 f$ H1 M. ?; w# G9 t/ C* E' Q2 _/ ~$ S/ V
* p* Z% @' ?/ ~3 ]HttpServletResponse.sendRedirect方法实现的请求重定向与RequestDispatcher.forward方法实现的请求转发的总结比较:
9 I0 g4 j* }4 T) P8 f* C
, l; o( _. r5 H& L! c* I% [/ M0 }3 L5 |# Z% ?' B7 D
2 K7 h" R9 N- N# v! O6 ]/ e 1.RequestDispatcher.forward方法只能将请求转发给同一个WEB应用中的组件;而HttpServletResponse.sendRedirect 方法不仅可以重定向到当前应用程序中的其他资源,7 X- d6 o! R) w" Q9 B4 D9 o& {
+ O! U5 A/ W3 t) @* Y7 M还可以重定向到同一个站点上的其他应用程序中的资源,甚至是使用绝对URL重定向到其他站点的资源。如果传递给HttpServletResponse.sendRedirect 方法的相对URL以“/”
6 Y. ]: ^! ]9 p, K2 ]' H: J- K2 t9 @ x2 s1 p# a& q- X7 ~
开头,它是相对于整个WEB站点的根目录;如果创建RequestDispatcher对象时指定的相对URL以“/”开头,它是相对于当前WEB应用程序的根目录。3 M, \& d3 K4 v' y8 f& W" l
' D& r8 Q K1 x* I: A2.调用HttpServletResponse.sendRedirect方法重定向的访问过程结束后,浏览器地址栏中显示的URL会发生改变,由初始的URL地址变成重定向的目标URL;而调用RequestDispatcher.forward
( Z5 ~$ i" a- P1 K: b$ e
; D, ]! t: e* ~" X' s2 r4 L, ]0 T方法的请求转发过程结束后,浏览器地址栏保持初始的URL地址不变。 ( G$ @2 B' L$ F" ?+ t/ r
0 {! W- u3 k0 a/ V' V% ^, R3.HttpServletResponse.sendRedirect方法对浏览器的请求直接作出响应,响应的结果就是告诉浏览器去重新发出对另外一个URL的访问请求,这个过程好比有个绰号叫“浏览器”的人写信找张三借钱,
" G# i. x8 G! c5 x. }2 W/ U; s- [# r) { @& L' c' |
张三回信说没有钱,让“浏览器“去找李四借,并将李四现在的通信地址告诉给了“浏览器”。于是,“浏览器”又按张三提供通信地址给李四写信借钱,李四收到信后就把钱汇给了“浏览器”。可见,“浏览器”
) R2 ^6 w1 u5 m5 |2 M3 A' h' d9 P+ l
: ~1 D, H/ }( c9 ~ ^: Y9 p- J3 f7 T' h一共发出了两封信和收到了两次回复,“浏览器”也知道他借到的钱出自李四之手。RequestDispatcher.forward方法在服务器端内部将请求转发给另外一个资源,浏览器只知道发出了请求并得到了响应结果,
0 v0 P5 z! x; M+ m* a9 O# Q
, [+ L1 `* E/ y2 h. [: h _并不知道在服务器程序内部发生了转发行为。这个过程好比绰号叫“浏览器”的人写信找张三借钱,张三没有钱,于是张三找李四借了一些钱,甚至还可以加上自己的一些钱,然后再将这些钱汇给了“浏览器”+ K. k4 `; X& o/ y: {" I) y
0 x+ s1 a& @) {1 x
。可见,“浏览器”只发出了一封信和收到了一次回复,他只知道从张三那里借到了钱,并不知道有一部分钱出自李四之手。! f, }9 S5 }" U0 p: x5 ]& ?3 L
; T; u* _) g; X$ ~ Z, y* [ U
0 g$ y' u( C+ g+ h5 U6 J8 r8 P# j/ W4.RequestDispatcher.forward方法的调用者与被调用者之间共享相同的request对象和response对象,它们属于同一个访问请求和响应过程;而HttpServletResponse.sendRedirect方法调用者与被调用者使用各自的
+ g0 ^0 n! p, Y3 Y5 y7 H; `9 P0 b7 G5 y1 e
request对象和response对象,它们属于两个独立的访问请求和响应过程。对于同一个WEB应用程序的内部资源之间的跳转,特别是跳转之前要对请求进行一些前期预处理,并要使用HttpServletRequest.setAttribute方法传递预处理* V/ i4 L( [) }6 E: H6 t3 ]
7 w! j/ h5 V2 L* F9 t: O
结果,那就应该使用RequestDispatcher.forward方法。不同WEB应用程序之间的重定向,特别是要重定向到另外一个WEB站点上的资源的情况,都应该使用HttpServletResponse.sendRedirect方法。! [- A6 \; X9 i* F2 f
/ \( K7 w% c: |. f
3 F$ ?( w3 ?+ Y; T/ F5.无论是RequestDispatcher.forward方法,还是HttpServletResponse.sendRedirect方法,在调用它们之前,都不能有内容已经被实际输出到了客户端。如果缓冲区中已经有
% P x4 ~$ O* A$ }' S- R
0 d4 _4 {" l$ p. w+ b6 K. Z了一些内容,这些内容将被从缓冲区中清除。+ B5 L( V' {* |& n. f
不要仅仅为了把变量传到下一个页面而使用session作用域,那会无故增大变量的作用域,; S) h3 s! _. Q& f( O
, P! k5 R( E9 Z( M
转发也许可以帮助你解决这个问题。6 c/ Z0 \- c& j! f3 u0 U( y& r
$ a9 C; W0 D# b; F, M. s' S
重定向:5 b* s) g+ y* T" W& h
以前的request中存放的变量全部失效,并进入一个新的request作用域。9 i+ H5 X) \5 ^* U
转发:) T' h& Q9 D! n- `
+ v; P7 a, K- p5 z
以前的request中存放的变量不会失效,就像把两个页面拼到了一起。
5 @$ L4 f, ?: y' E" {+ p
% C2 ~+ [( H0 ]8 N0 W
D P' H- L- b; V* P) h
9 j9 Y& m+ X+ A9 n6 t$ u: d |
|