【42】
请求转发:
问题:服务器在接收到浏览器的请求后,仅仅使用一个Servlet 进行请求处理,会造成不同的 Servlet 逻辑代码冗余,Servlet 的职责不明确。
解决:使用请求转发。
特点:
- 一次请求
- 地址栏信息不改变。
- Request 对象作用域
问题:使用请求转发后,不同的 Servlet 之间怎么进行数据的共享呢?或者说数据怎么从一个 servlet 流转给另外一个 Servlet 呢?
解决:使用 request 对象的作用域
使用:
request.setAttribute(object name,Object value);
request.getAttribute(Object obj)
作用:解决了一次请求内的不同 Servlet 的数据(请求数据+其他数据)共享问题。
作用域:基于请求转发,一次请求中的所有 Servlet 共享。
注意:使用 Request 对象进行数据流转,数据只在一次请求内有效。
特点:
- 服务器创建
- 每次请求都会创建
- 生命周期一次请求
重定向
问题:如果当前的请求,Servlet 无法进行处理怎么办?如果使用请求转发,造成表单数据重复提交怎么办?
解决:使用重定向
使用:response.sendRedirect(“路径”).
本地路径为:uri
网络路径为:定向资源的 URL 信息
特点:
- 两次请求
- 浏览器地址栏信息改变
- 避免表单重复提交
转发和重定向的区别:
Cookie
测试代码:MyServlet1
public class MyServlet1 extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
//设置请求编码格式
req.setCharacterEncoding("utf-8");
//设置响应编码格式
resp.setContentType("text/html;charset=utf-8");
//获取请求信息
String a=req.getParameter("a");
String b=req.getParameter("b");
//处理请求信息
System.out.println(a+":"+b);
//响应处理结果
//创建Cookie数据
Cookie c=new Cookie("b", b);
//设置Cookie信息
c.setMaxAge(3600*24*3);//设置cookie的有效期。
//设置有效路径
c.setPath("/cookie/my2");
//响应Cookie数据
resp.addCookie(c);
//直接响应
resp.getWriter().write("我是MyServlet1,处理完毕");
//请求转发
//重定向
}
}
MyServlet2:
public class MyServlet2 extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
//设置请求编码格式
req.setCharacterEncoding("utf-8");
//设置响应编码格式
resp.setContentType("text/html;charset=utf-8");
//获取请求信息
//String b=req.getParameter("b");
String bc=null;
//获取Cookie数据
Cookie[] cookies = req.getCookies();
for(Cookie ck:cookies){
if("b".equals(ck.getName())){
bc=ck.getValue();
}
}
String c=req.getParameter("c");
//处理请求信息
System.out.println(bc+":"+c);
//响应处理结果
//直接响应
resp.getWriter().write("我是MyServlet2,处理完毕");
//请求转发
//重定向
}
}
Session
问题:Request 对象解决了一次请求内的不同 Servlet 的数据共享问题,那么一个用户的不同请求的处理需要使用相同的数据怎么办呢?
解决:使用 session 技术。
原理:用户使用浏览器第一次向服务器发送请求,服务器在接受到请求后,调用对应的 Servlet 进行处理。在处理过程中会给用户创建一个 session 对象,用来存储用户请求处理相关的公共数据,并将此 session 对象的 JSESSIONID 以 Cookie 的形式存储在浏览器中(临
时存储,浏览器关闭即失效)。用户在发起第二次请求及后续请求时,请求信息中会附带 JSESSIONID,服务器在接收到请求后,调用对应的 Servlet 进行请求处理,同时根据 JSESSIONID 返回其对应的 session 对象。
特点:
- Session 技术是依赖 Cookie 技术的服务器端的数据存储技术。
- 由服务器进行创建
- 每个用户独立拥有一个 session
- 默认存储时间为 30 分钟
作用:
解决了一个用户的不同请求的数据共享问题。
使用:
- 创建 Session 对象
- 存储数据到 session 对象
- 获取 session 对象
- 获取数据从 session 对象
- 如果获取 session 中不存在的数据返回 null。
注意:只要不关闭浏览器,并且 session 不失效的情况下,同一个用户的任意请求在项目的任意Servlet中获取到的都是同一个session对象。
作用域:一次会话
session学习:
session的使用:
//创建session对象
HttpSession session =req.getSession();
//存储数据到session中
session.setAttribute(String name, Object value);
//获取session对象
HttpSession session =req.getSession();
//获取session中的数据
session.getAttribute(String uname);注意:返回的object类型,需要强制转换
//删除session中的数据
session.removeAttribute(String uname);
//注意:如果有数据则删除,没有则什么都不做。
流程: 1、浏览器发起请求到Aservlet,在AServlet中使用req.getSession()获取Session对象,如果此次请求中没有SessionID则创建一个新的Session对象,如果有SessionID则将其对应的Session对象返回(前提是该session对象没有到期),如果session对象到期销毁了,就算有sessionID也会重新创建一个Session。
2、校验session是否失效,存储数据到session对象中或者获取session中的数据或者删除session中的数据
特点:session解决了同一个用户不同请求 的数据共享问题。
session的作用域:浏览器不关闭,session不失效,则同一用户的任意请求获取的都是同一个session 一次会话
session的设置:
session默认有效时间为30分钟,可以在tomcat下的web.xml中进行配置
注意:此种配置方式是所有的tomcat下的项目默认为30分钟
也可以在代码中使用
session.setMaxInactiveInterval(int seconds);//设置session的有效时间,参数为整数类型的秒
session.invalidate();//强制销毁session
测试代码:
UserServlet
public class UserServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
//设置请求编码格式
req.setCharacterEncoding("utf-8");
//设置响应编码格式
resp.setContentType("text/html;charset=utf-8");
//获取请求信息
String uname=req.getParameter("uname");
String pwd=req.getParameter("pwd");
//处理请求信息
//校验用户信息
if("张三".equals(uname) && "123".equals(pwd)){
//登录成功
//响应处理结果
}else{
//登录失败
//创建Session并增加登录失败标记
HttpSession session = req.getSession();
session.setAttribute("flag","loginFalse");
//响应处理结果(重定向到登录页面)
resp.sendRedirect("login");
}
}
}
LoginServlet
public class LoginServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
//设置请求编码格式
req.setCharacterEncoding("utf-8");
//设置响应编码格式
resp.setContentType("text/html;charset=utf-8");
//获取Session中的数据
HttpSession session = req.getSession();
String str=(session.getAttribute("flag")==null?"":"用户名或密码错误");
//销毁session
session.invalidate();
//获取请求信息
//处理请求信息
//响应处理结果
//直接响应
resp.getWriter().write("<html>");
resp.getWriter().write("<head>");
resp.getWriter().write("</head>");
resp.getWriter().write("<body>");
resp.getWriter().write("<form action='user' method='get'>");
resp.getWriter().write("<font color='red'>"+str+"</font>");
resp.getWriter().write(" 用户名:<input type='text' name='uname' value=''/><br />");
resp.getWriter().write(" 密码:<input type='password' name='pwd' value=''/><br />");
resp.getWriter().write("<input type='submit' value='登录'/><br />");
resp.getWriter().write("</form>");
resp.getWriter().write("</body>");
resp.getWriter().write("</html>");
}
}
ServletContext 对象:
作用:解决了不同用户之间的数据共享问题
使用:创建ServletContext对象
ServletContext sc1 = this.getServletContext();
ServletContext sc2 =this.getServletConfig().getServletContext();
ServletContext sc3=req.getSession().getServletContext();
存储用户共享数据 sc.setAttribute(String name,Object value);
获取用户共享数据 sc.getAttribute(String name);
删除共享数据 sc.removeAttribute(String name);
获取web.xml中的全局配置属性,作用:将部分动作和源文件进行解耦,我们只需要在xml配置文件中进行相关配置就会改变代码的执行效果。
sc.getInitParameter(String name);注意返回的是String类型数据
获取webRoot下的资源流对象
sc.getResourceAsStream(String path);
注意:path为相对路径,写从webRoot开始查找资源的路径即可。
获取webRoot下的资源绝对路径 sc.getRealPath(String path);
注意:path为相对路径,写从webRoot开始查找资源的路径即可。
特点:服务器创建,被所有用户共享
生命周期:服务器开启到服务器关闭
ServletConfig 对象:
问题:使用 ServletContext 对象可以获取 web.xml 中的全局配置文件,在 web.xml 中每个 Servlet 也可以进行单独的配置,那么该怎么获取配置信息呢?
解决:使用 ServletConfig 对象
作用:ServletConfig 对象是 Servlet 的专属配置对象,每个 Servlet 都
单独拥有一个 ServletConfig 对象,用来获取 web.xml 中的配置信息。
使用:a、获取 ServletConfig 对象 b、 获取 web.xml 中 servlet 的配置信息