会话及会话技术

会话

会话指的是的是一个客户端服务器之间连续发生的一系列请求和响应过程,例如,一个用户在某网站上的整个购物过程就是一个会话。

当用户通过浏览器访问Web应用时,通常情况下,服务器需要对用户的状态进行跟踪。例如,用户在网站结算商品时,Web服务器必须根据请求用户的身份,找到该用户所购买的商品。在Web开发中,服务器跟踪用户信息的技术称为会话技术。在Servlet技术中,提供了两个用于保存会话数据的对象,分别是客户端会话技术:Cookie和服务器端会话技术:Session。

它是客户端浏览器的缓存文件,里面记录了客户浏览器访问网站的一些内容。同时,也是HTTP协议请求和响应消息头的一部分。

在现实生活中,当顾客在购物时,商城经常会赠送顾客一张会员卡,卡上记录用户的个人信息、消费额度和积分额度等。顾客一旦接受了会员卡,以后每次光临该商场时,都可以使用这张会员卡,商场也将根据会员卡上的消费记录计算会员的优惠额度和累加积分。在Web应用中,Cookie的功能类似于这张会员卡,当用户通过浏览器访问Web服务器时,服务器会给客户端发送一些信息,这些信息都保存在 Cookie 中。这样,当该浏览器再次访问服务器时,都会在请求头中将 Cookie 发送给服务器,方便服务器对浏览器做出正确的响应。

cookie作用

  1. cookie一般用于存储少量的不太敏感的数据
  2. 在不登录的情况下,完成服务器对客户端的身份识别

cookie的属性

属性名称 属性作用 是否重要
name cookie的名称 必要属性
value cookie的值(不能是中文) 必要属性
path cookie的路径 重要
domain cookie的域名 重要
maxAge cookie的生存时间。 重要
version cookie的版本号。 不重要
comment cookie的说明。 不重要

使用步骤

  1. 创建Cookie对象,绑定数据:new Cookie(String name, String value)
  2. 发送Cookie对象:resp.addCookie(Cookie cookie)
  3. 获取Cookie,拿到数据:Cookie[] request.getCookies()
    然后遍历,使用getValue()
  4. Cookie常用方法
    会话及会话技术

细节

Cookie有大小,个数限制。每个网站最多只能存20个cookie,且大小不能超过4kb。同时,所有网站的cookie总数不超过300个。

当删除Cookie时,设置maxAge值为0。当不设置maxAge时,使用的是浏览器的内存,当关闭浏览器之后,cookie将丢失。设置了此值,就会保存成缓存文件(值必须是大于0的,以秒为单位)。设置cookie保存时间为一个月:setMaxAge(60*60*24*30)

  1. 正数:指定cookie存货的秒数
  2. 负数:默认值
  3. 0:删除cookie信息

cookie能不能存中文:tomcat7之前不能,tomcat8之后可以,但还是不支持特殊字符,建议使用URL编码。

cookie共享问题:默认情况:假设在一个tomcat服务器中,部署了多个web项目,这些项目的cookie不能共享
2. 不同项目之间共享:使用setPath(String path),设置cookie的获取范围。默认情况下,设置当前的虚拟目录,只有一个项目下的资源才可以共享cookie。如果要在不同项目间共享,可以将path设置为"/"
3. 不同tomcat服务器间共享:setDomain(String path),如果设置一级域名相同,那么多个服务器之间cookie可以共享。如:setDomain(".baidu.com"),那么tieba.baidu.com和news.baidu.com中cookie可以共享

案例:记住上一次访问时间

/**
 在服务器中的Servlet判断是否有一个名为lastTime的cookie
 1. 有:不是第一次访问
 1. 响应数据:欢迎回来,您上次访问时间为:2018年6月10日11:50:20
 2. 写回Cookie:lastTime=2018年6月10日11:50:01
 2. 没有:是第一次访问
 1. 响应数据:您好,欢迎您首次访问
 2. 写回Cookie:lastTime=2018年6月10日11:50:01
其实使用 session 的 getLastAccessedTime()更简单
 */

@WebServlet("/cookieTest")
public class CookieTest extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //设置响应的消息体的数据格式以及编码
        response.setContentType("text/html;charset=utf-8");

        //1.获取所有Cookie
        Cookie[] cookies = request.getCookies();
        boolean flag = false;//没有cookie为lastTime
        //2.遍历cookie数组
        if(cookies != null && cookies.length > 0){
            for (Cookie cookie : cookies) {
                //3.获取cookie的名称
                String name = cookie.getName();
                //4.判断名称是否是:lastTime
                if("lastTime".equals(name)){
                    //有该Cookie,不是第一次访问

                    flag = true;//有lastTime的cookie

                    //设置Cookie的value
                    //获取当前时间的字符串,重新设置Cookie的值,重新发送cookie
                    Date date  = new Date();
                    SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss");
                    String str_date = sdf.format(date);
                    System.out.println("编码前:"+str_date);
                    //URL编码
                    str_date = URLEncoder.encode(str_date,"utf-8");
                    System.out.println("编码后:"+str_date);
                    cookie.setValue(str_date);
                    //设置cookie的存活时间
                    cookie.setMaxAge(60 * 60 * 24 * 30);//一个月
                    response.addCookie(cookie);

                    //响应数据
                    //获取Cookie的value,时间
                    String value = cookie.getValue();
                    System.out.println("解码前:"+value);
                    //URL解码:
                    value = URLDecoder.decode(value,"utf-8");
                    System.out.println("解码后:"+value);
                    response.getWriter().write("<h1>欢迎回来,您上次访问时间为:"+value+"</h1>");

                    break;

                }
            }
        }

        if(cookies == null || cookies.length == 0 || flag == false){
            //没有,第一次访问

            //设置Cookie的value
            //获取当前时间的字符串,重新设置Cookie的值,重新发送cookie
            Date date  = new Date();
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss");
            String str_date = sdf.format(date);
            System.out.println("编码前:"+str_date);
            //URL编码
            str_date = URLEncoder.encode(str_date,"utf-8");
            System.out.println("编码后:"+str_date);

            Cookie cookie = new Cookie("lastTime",str_date);
            //设置cookie的存活时间
            cookie.setMaxAge(60 * 60 * 24 * 30);//一个月
            response.addCookie(cookie);

            response.getWriter().write("<h1>您好,欢迎您首次访问</h1>");
        }


    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request, response);
    }
}

Session

Cookie 技术可以将用户的信息保存在各自的浏览器中,并且可以在多次请求下实现数据的共享。但是,如果传递的信息比较多,使用Cookie技术显然会增大服务器端程序处理的难度,这时,可以使用Session技术。注意Session技术是cookie的拓展,建立在cookie的基础上。

同时Session也是Servlet规范中四大域对象之一的会话域对象。

使用

  1. 获取HttpSession对象:HttpSession session = request.getSession();
  2. 使用HttpSession对象:
    Object getAttribute(String name)
    void setAttribute(String name, Object value)
    void removeAttribute(String name)
  3. 销毁Session
    void invalidate()

原理:当浏览器访问Web服务器时,Servlet容器就会创建一个Session对象和ID属性,将ID属性通过Cookie发送给客户端。当客户端后续访问服务器时,只要将ID传递给服务器,服务器就能判断出该请求是哪个客户端发送的,从而选择与之对应的Session对象为其服务。

细节

  1. 当客户端关闭后,服务器不关闭,两次获取session是否为同一个?
    默认情况下不是。如果需要相同,设置Cookie存活时间,让Cookie持久化保存即可。

    HttpSession session = req.getSession();
    Cookie c = new Cookie("JSESSIONID",seesion.getId());
    c.setMaxAge(60*60);
    resp.addCookie(c);
    
  2. 客户端不关闭,服务器关闭后,两次获取的session是同一个吗?
    不是同一个。

设置会话生命周期

  1. long getLastAccessedTime():返回客户端最后一次请求会话的时间。返回值是毫秒数,需要手动转换成Date类
  2. getMaxInactiveInterval():以秒为单位返回一个会话内两个请求最大时间间隔。
  3. setMaxInactiveInterval():以秒为单位设置session的有效时间。

钝化和活化

把长时间不用,但还不到过期时间的HttpSession进行序列化,写到磁盘上。成为钝化。

什么时候使用钝化

第一种情况:当访问量很大时,服务器会根据getLastAccessTime来进行排序,对长时间不用,但是还没到过期时间的HttpSession进行持久化。

第二种情况:当服务器进行重启的时候,为了保持客户HttpSession中的数据,也要对HttpSession进行持久化

注意

HttpSession的持久化由服务器来负责管理,我们不用关心。

只有实现了序列化接口的类才能被序列化,否则不行。

会话及会话技术

上一篇:Join方法


下一篇:surface 主机不能打开Microsoft商店,错误代码0x80131500