Servlet的跳转和重定向也算是一种处理页面间转向的机制(之前Nginx挖的坑还没填,HTTPS配置完想设置重定向。然后莫名其妙的就配置不了,各种404Not Found==Https大势所趋所以不出三个月一定搞定)。而Servlet设置页面跳转以及重定向的初衷也是为了省去在doGet()方法亦或者是Service方法当中再去一步步写原来就有的页面作为响应体,将这个过程包装之后也就出现了重定向和跳转。
服务器内部跳转
第一种情况:
服务器内部跳转到一个页面中
例如:
String url = "/hello.html";
//获得一个指向url的跳转对象
RequestDispatcher dispatcher = request.getRequestDispatcher(url);
//进行跳转 跳转的同时把request和response转发给要跳转到的url
dispatcher.forward(request, response);
或者:
String url = "/hello.html";
request.getRequestDispatcher(url).forward(request, response);
第二种情况:
服务器内部跳转到一个servlet中
String url = "/TestServlet";
request.getRequestDispatcher(url).forward(request, response);
服务器内部跳转的特点:
1.需要使用request对象来完成
2.服务器内部跳转期间,其实就是把请求request和响应response转发到一下资源中,所以在整个跳转期间所有涉及到的资源使用的都是同一个request和response。(我们可以利用这个特点将来在多个资源之间进行数据的传递)
3.服务器内部跳转,【不会】改变浏览器地址栏中的地址
客户端重定向
第一种情况:
客户端重定向到一个页面中
String url = "hello.html";
response.sendRedirect(url);
第二种情况:
客户端重定向到一个servlet中
String url = "TestServlet";
response.sendRedirect(url);
客户端重定向的特点:
1.需要使用response对象来完成
2.每一个客户端重定向,浏览器都会发出新请求,也就意味着在服务器内部会产生新的request对象和response对象
3.客户端重定向,【会】改变浏览器地址栏中的地址
以上两者直间的区别:
跳转
1.使用request完成
2.跳转的本质是在服务器内部把请求和响应进行转发,转发给下一个资源
3.如果request中有数据,在跳转到的每一个资源中都可以拿到一个数据(从开始到结束只有一个request对象)
4.由于服务器跳转是把请求和响应在服务器内部中来回转发,所以浏览器的地址栏中的地址是不受影响的,所以始终还是第一次发出请求的地址
5.在设置跳转的资源地址的时候,路径的前面一般要加上/
例如 String path = "/a.html";
重定向
1.使用response完成
2.重定向的本质是把资源路径通过响应返回给浏览器(通过响应头信息),让浏览器向这个新地址发送一个新请求
3.如果request中有数据,重定向后在新的资源中是拿不到这个数据的(重定向会发一个全新的请求,但是数据在上一个老的请求中)
4.由于重定向让浏览器发出新的请求,所以浏览器地址栏中的地址会变成新请求的地址
5.在设置重定向的资源地址的时候,路径的前面一般不加/
例如 String path = "a.html";
但是如果一定要加/的话,那么还需要额外加上一个项目名
例如 String path = "/web_test/a.html";
同时也可以动态获取项目:req.getContextPath()
String path = req.getContextPath()+"/a.html";
Demo:
package xin.string;
import java.io.IOException;
import javax.servlet.RequestDispatcher;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@WebServlet("/checkname")
public class CheckByname extends HttpServlet{
/**
*
*/
private static final long serialVersionUID = 1L;
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
// TODO Auto-generated method stub
String url = "/failed.html";
String servletUrl = "/checkpass";
if(req.getParameter("username").equals("Tom")){
req.getRequestDispatcher(servletUrl).forward(req, resp);
}else{
RequestDispatcher dispatcher = req.getRequestDispatcher(url);
dispatcher.forward(req, resp);
}
}
}
package xin.string;
import java.io.IOException;
import javax.servlet.RequestDispatcher;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@WebServlet("/checkpass")
public class CheckBypasswrod extends HttpServlet{
/**
*
*/
private static final long serialVersionUID = 1L;
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
// TODO Auto-generated method stub
String url = "/hello.html";
String failed = "error.html";
if(req.getParameter("password").equals("123456")){
RequestDispatcher dispatcher = req.getRequestDispatcher(url);
dispatcher.forward(req, resp);
}else{
resp.sendRedirect(failed);
}
}
}
以上两个程序完成了服务器内部跳转以及客户端重定向这些操作。