import java.text.*; import java.util.*; import java.io.*; import javax.servlet.http.*; import javax.servlet.*; import com.bjpowernode.exam.model.*; import com.bjpowernode.exam.manager.*; public class SearchStudentServlet extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String sBeginDate = request.getParameter("beginDate"); String sEndDate = request.getParameter("endDate"); Date beginDate = new Date(); Date endDate = new Date(); try { beginDate = new SimpleDateFormat("yyyy-MM-dd").parse(sBeginDate); endDate = new SimpleDateFormat("yyyy-MM-dd").parse(sEndDate); }catch(Exception e) { e.printStackTrace(); } StudentManager studentManager = new StudentManagerImpl(); List<Student> studentList = studentManager.findStudentList(beginDate, endDate); //將學生列表設置到requet范圍中 //request.setAttribute("student_list", studentList); //轉發,轉發是在服務器端轉發的,客戶端是不知道的 //request.getRequestDispatcher("/student_list.jsp").forward(request, response); //將studentList放到session中 HttpSession session = request.getSession(); session.setAttribute("student_list", studentList); //重定向,不會共享request //以下寫法錯誤,該 "/"代表了8080端口 //response.sendRedirect("/student_list.jsp"); response.sendRedirect(request.getContextPath() + "/student_list.jsp"); } }
這個里面嘗試了兩種調到后面的Jsp方法,在servlet中調用轉發、重定向的語句如下:
實現轉發:
//轉發,轉發是在服務器端轉發的,客戶端是不知道的 request.getRequestDispatcher("/student_list.jsp").forward(request, response);
分析:請求轉發是服務器內部把對一個request/response的處理權,移交給另外一個對於客戶端而言,它只知道自己最早請求的那個A,而不知道中間的B,甚至C、D。 傳輸的信息不會丟失。
實現重定向:
//重定向,不會共享request //以下寫法錯誤,該 "/"代表了8080端口 response.sendRedirect("/student_list.jsp"); response.sendRedirect(request.getContextPath() + "/student_list.jsp");
深入(分析理解)
轉發過程
客戶首先發送一個請求到服務器端,服務器端發現匹配的servlet,並指定它去執行,當這個servlet執行完之后,它要調用getRequestDispacther()方法,把請求轉發給指定的student_list.jsp,整個流程都是在服務器端完成的,而且是在同一個請求里面完成的,因此servlet和jsp共享的是同一個request,在servlet里面放的所有東西,在student_list中都能取出來,因此,student_list能把結果getAttribute()出來,getAttribute()出來后執行完把結果返回給客戶端。整個過程是一個請求,一個響應。
重定向過程
客戶發送一個請求到服務器,服務器匹配servlet,這都和請求轉發一樣,servlet處理完之后調用了sendRedirect()這個方法,這個方法是response的方法,所以,當這個servlet處理完之后,看到response.senRedirect()方法,立即向客戶端返回這個響應,響應行告訴客戶端你必須要再發送一個請求,去訪問student_list.jsp,緊接着客戶端受到這個請求后,立刻發出一個新的請求,去請求student_list.jsp,這里兩個請求互不干擾,相互獨立,在前面request里面setAttribute()的任何東西,在后面的request里面都獲得不了。可見,在sendRedirect()里面是兩個請求,兩個響應。
淺出(表象)
轉發
當用RequestDispatcher請求轉發后,地址欄為http://localhost:8080/test/TestServlet
這真好應正了上面的分析,我們起初請求的就一個servlet,至於你服務器端怎么轉,流程怎么樣的,我客戶端根本就不知道,我發了請求后我就等着響應,那你服務器那邊願意怎么轉就怎么轉,我客戶端不關心也沒法知道,所以當服務器端轉發到jsp后,它把結果返回給客戶端,客戶端根本就不知道你這個結果是我真正訪問的servlet產生的,還是由servlet轉發后下一個組件產生的。
重定向
當用sendRedirect重定向后,地址欄為http://localhost:8080/test/student_list.jsp
因為這個時候,客戶端已經知道了他第二次請求的是student_list.jsp,服務器已經告訴客戶端要去訪問student_list.jsp了,所以地址欄里會顯示想要訪問的結果。
總結
轉發在服務器端完成的;重定向是在客戶端完成的
轉發的速度快;重定向速度慢
轉發的是同一次請求;重定向是兩次不同請求
轉發不會執行轉發后的代碼;重定向會執行重定向之后的代碼
轉發地址欄沒有變化;重定向地址欄有變化
轉發必須是在同一台服務器下完成;重定向可以在不同的服務器下完成
Forward是在服務器端的跳轉,就是客戶端一個請求發給服務器,服務器直接將請求相關的參數的信息原封不動的傳遞到該服務器的其他jsp或servlet去處理,而sendredirect是在客戶端的跳轉,服務器會返回給客戶端一個響應報頭和新的URL地址,原來的參數什么的信息如果服務器端沒有特別處理就不存在了,瀏覽器會訪問新的URL所指向的servlet或jsp,這可能不是原先服務器上的webservce了。
原文連接:深入淺出Java重定向和請求轉發