Cookie & Session

会话技术

# 一次会话中包含多次请求和多次响应
  * 一次会话:浏览器第一次给服务器资源发送请求,会话建立;直到有一方断开,会话终止。

# 功能:在一次会话的范围内的多次请求和响应间,实现数据的共享

# 种类:
  * 客户端会话技术:Cookie
  * 服务器端会话技术:Session

Cookie

快速入门

# 概念:客户端会话技术,将数据保存到客户端

# 使用步骤
  * 创建 Cookie对象,绑定数据:new Cookie(String name, String value)
  * 发送 Cookie对象:response.addCookie(Cookie cookie)
  * 获取 Cookie对象,拿到数据:Cookie[ ] request.getCookies()

package cn.itcast.cookie;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * Cookie快速入门
 */
@WebServlet("/cookieDemo1")
public class cookieDemo1 extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 创建 Cookie 对象
        Cookie c = new Cookie("msg","hello");
        // 发送 Cookie
        response.addCookie(c);
    }

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request, response);
    }
}
View Code
package cn.itcast.cookie;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * Cookie快速入门
 */
@WebServlet("/cookieDemo2")
public class cookieDemo2 extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 获取Cookie
        Cookie[] cs = request.getCookies();
        // 遍历数组,获取数据
        if (cs != null) {
            for (Cookie c : cs) {
                String name = c.getName();
                String value = c.getValue();

                System.out.println(name + "..." + value);
            }
        }
    }

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

# 实现原理:
  * 基于响应头 set-cookie 请求头 cookie 实现

Cookie的细节

# 一次可不可以发送多个 Cookie
  * 可以,创建多个 Cookie对象,使用 response调用多次 addCookie() 方法发送即可

# Cookie在浏览器中保存多长时间
  * 默认情况下,当浏览器关闭后,Cookie数据被销毁
  * 持久化存储:setMaxAge(int seconds)
    * 正数:将 Cookie数据写到硬盘的文件中,持久化存储。并且指定 Cookie存活时间,时间到后,自动失效
    * 负数默认值,浏览器关闭后删除cookie
    * 删除 Cookie信息

package cn.itcast.cookie;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * Cookie快速入门
 */
@WebServlet("/cookieDemo4")
public class cookieDemo4 extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 创建 Cookie 对象
        Cookie c1 = new Cookie("msg","setMaxAge");
        // 设置Cookie的存活时间,30s后自动删除
        c1.setMaxAge(30);
        // 发送 Cookie
        response.addCookie(c1);

    }

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

# Cookie能不能存储中文
  * 在 Tomcat 8 之前 Cookie中不能直接存储中文数据
    * 需要将中文数据转码,一般采用 URL编码 (%3)
  * 在 Tomcat 8 之后,Cookie支持中文数据,但是不支持特殊字符,建议还是使用URL编码存储和解析

# Cookie共享问题
  1)同一个 Tomcat服务器中,不同 web项目间,Cookie共享问题?
    * 默认情况下 Cookie不能共享
    * setPath(String path):设置 Cookie的获取范围,默认情况下为当前的虚拟目录
      * 如果要共享,则可以将 path设置为  " / "

  2)不同的 Tomcat服务器间,Cookie共享问题
    * setDomain(String path):如果设置一级域名相同,那么多个服务器之间 Cookie可以共享
      * 例如:setDomain(".baidu.com"),那么 tieba.baidu.com 和 news.baidu.com 中的 cookie可以共享

Cookie的特点和作用

# 特点
  * Cookie存储数据在客户端浏览器
  * 浏览器对于单个 Cookie 的大小有限制(4kb)以及对同一个域名下的总的 Cookie数量也有限制(20个)

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

案例

# 需求
  1)如果是第一次访问,则提示:您好,欢迎您的首次访问
  2)如果不是第一次访问,则提示:欢迎回来,您上次的访问时间为:${时间字符串}$

# 分析
  1)可以采用 Cookie来完成
  2)在服务器中的 Servlet判断是否有一个名为 lastTime 的 Cookie
    * 有:不是第一次访问
      ① 响应数据:欢迎回来:您上次访问的时间为:2020年8月31日16:37:40
      ② 写回 Cookie:lastTime = 2020年8月31日16:40:00
    * 没有:是第一次访问
      ① 响应数据:您好,欢迎首次访问
      ② 写回 Cookie:2020年8月31日16:37:40

# 代码实现

package cn.itcast.cookie;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.net.URLDecoder;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.net.URLEncoder;

@WebServlet("/CookieTest")
public class CookieTest extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 设置响应的消息体的数据格式及编码
        response.setContentType("text/html;charset=utf-8");
        // 获取所有的Cookie
        Cookie[] cookies = request.getCookies();
        boolean flag = false;
        // 遍历Cookie数组
        if (cookies != null && cookies.length > 0) {
            for (Cookie cookie : cookies) {
                // 获取cookie的名称
                String name = cookie.getName();
                // 判断名称是否是 lastTime
                if ("lastTime".equals(name)) {
                    // 有该cookie,不是第一次访问
                    flag = true;
                    // 获取cookie的value
                    String value = URLDecoder.decode(cookie.getValue(),"utf-8");
                    // 设置cookie的value
                    Date date = new Date();
                    SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss");
                    String str_date = URLEncoder.encode(sdf.format(date),"utf-8");
                    cookie.setValue(str_date);
                    // 设置cookie的存储时间
                    cookie.setMaxAge(60 * 60 * 24 * 30);
                    response.addCookie(cookie);
                    response.getWriter().write("<h1>欢迎回来,您上次访问时间为:" + value + "</h1>");

                    break;
                }
            }
        }

        if (cookies == null || cookies.length == 0 || flag == false) {
            // 第一次访问
            Date date = new Date();
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss");
            String str_date = URLEncoder.encode(sdf.format(date),"utf-8");
            Cookie cookie = new Cookie("lastTime", str_date);
            cookie.setValue(str_date);
            // 设置cookie的存储时间
            cookie.setMaxAge(60 * 60 * 24 * 30);
            response.addCookie(cookie);
            response.getWriter().write("<h1>您好,欢迎首次访问</h1>");
        }
    }

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

JSP

入门学习

# 概念:Java Server Pages:Java服务器端页面
  * 一个特殊的页面,其中既可以定义 HTML标签,也可以定义 Java代码
  * 可以简化书写

# 原理:JSP本质上就是一个 Servlet

# JSP脚本:JSP定义代码的方式
  * <%   代码 %>:定义的 Java代码 在 service() 方法中
  * <%!  代码 %>:定义的 Java代码 在JSP跳转后的 Java类的成员位置
  * <%= 代码 %>:定义的 Java代码 输出到页面上

<%--
  Created by IntelliJ IDEA.
  User: DELL
  Date: 2020/8/12
  Time: 20:16
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>$Title$</title>
</head>
<body>
<%
    System.out.println("hello jsp!");
    String contextPath = request.getContextPath();
%>

<%!
    int i = 3;
%>

<%= i %>


<h1>hi,jsp</h1>

</body>
</html>
View Code 

# JSP的内置对象
  * 在 JSP页面中不需要获取和创建对象,可以直接使用
  * 重要的 3个对象:
    1)request:代表请求
    2)response:代表响应
    3)out:字符输出流对象,可以将数据输出到页面上,也 response.getWriter() 类型
      * 区别:在 Tomcat服务器给客户端作出响应前,会先找 response缓冲区数据,再找 out缓冲区数据,因此 getWriter() 数据输出永远在 out 前

# 案例:改造 Cookie案例

<%@ page import="java.net.URLDecoder" %>
<%@ page import="java.util.Date" %>
<%@ page import="java.text.SimpleDateFormat" %>
<%@ page import="java.net.URLEncoder" %>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>itcast</title>
</head>
<body>
<%
    // 获取所有的Cookie
    Cookie[] cookies = request.getCookies();
    boolean flag = false;

    // 遍历Cookie数组
    if (cookies != null && cookies.length > 0) {
        for (Cookie cookie : cookies) {
            // 获取cookie的名称
            String name = cookie.getName();
            // 判断名称是否是 lastTime
            if ("lastTime".equals(name)) {
                flag = true;
                // 获取cookie的value
                String value = URLDecoder.decode(cookie.getValue(), "utf-8");
                // 设置cookie的value
                Date date = new Date();
                SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss");
                String str_date = URLEncoder.encode(sdf.format(date), "utf-8");
                cookie.setValue(str_date);
                // 设置cookie的存储时间
                cookie.setMaxAge(60 * 60 * 24 * 30);
                response.addCookie(cookie);
%>

<h1>欢迎回来,您上次访问时间为:<%=value%>></h1>

<%
                break;
            }
        }
    }

    if (cookies == null || cookies.length == 0 || flag == false) {
        // 第一次访问
        Date date = new Date();
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss");
        String str_date = URLEncoder.encode(sdf.format(date), "utf-8");
        Cookie cookie = new Cookie("lastTime", str_date);
        cookie.setValue(str_date);
        // 设置cookie的存储时间
        cookie.setMaxAge(60 * 60 * 24 * 30);
        response.addCookie(cookie);
%>

<h1>您好,欢迎首次访问</h1>

<%
    }
%>
</body>
</html>
View Code 

Session

快速入门

# 概念:服务器端会话技术,在一次会话的多次请求间共享数据,将数据保存在服务器端的对象中

# HttpSession对象
  * 获取 HttpSession:HttpSession session = request.getSession()
  * 使用 HttpSession:Object getAttribute(String name),void setAttribute(String name, Object value),void removeAttribute(String name)

# 原理:Session的实现依赖于 Cookie

package cn.itcast.session;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;

@WebServlet("/sessionDemo1")
public class SessionDemo1 extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 使用 session 共享数据
        HttpSession session = request.getSession();
        // 存储数据
        session.setAttribute("msg","hello session...");

    }

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request, response);
    }
}
View Code
package cn.itcast.session;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;

@WebServlet("/sessionDemo2")
public class SessionDemo2 extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 使用 session 获取数据
        HttpSession session = request.getSession();
        // 获取数据
        Object msg = session.getAttribute("msg");
        System.out.println(msg);

    }

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

细节

# 客户端关闭后,服务器端不关闭,两次获取的 Session是否为同一个?
  * 默认情况下,不是
  * 如果要相同,可以创建 Cookie,键指定为 JSESSIONID,设置最大存活时间,让 Cookie持久化保存

package cn.itcast.session;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.*;
import java.io.IOException;

@WebServlet("/sessionDemo3")
public class SessionDemo3 extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 使用 session 获取数据
        HttpSession session = request.getSession();
        // 期望客户端关闭后,session也能相同
        Cookie cookie = new Cookie("JSESSIONID", session.getId());
        cookie.setMaxAge(60 * 60);
        response.addCookie(cookie);
        System.out.println(session);
    }

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

# 客户端不关闭,服务器端关闭,两次获取的 Session是否为同一个?
  * 不是同一个,但是为了数据不丢失,Tomcat 自动完成以下工作:
    * Session的钝化:在服务器正常关闭之前,将 Session对象序列化存储到硬盘上
    * Session的活化:在服务器启动后,将 Session文件转化为内存中的 Session对象

# Session什么时候被销毁
  1)服务器被关闭
  2)Session对象调用 invalidate() 方法
  3)Session默认失效时间:30分钟
    * 可以在web.xml <session-config>下的<session-timeout>标签内设置

# Session的特点
  * Session用于存储一次会话的多次请求的数据,存在服务器端
  * Session可以存储任意类型,任意大小的数据

# Session 和 Cookie的区别
  1)Session存储数据在服务器端,Cookie存储在客户端
  2)Session没有数据大小的限制,Cookie有
  3)Session数据安全,Cookie相对不安全

案例

# 案例需求
  1)访问带有验证码的登陆页面 login.html
  2)用户输入用户名,密码和验证码
    * 如果用户名或密码有误,跳转登录页面,提示:用户名或密码有误
    * 如果验证码输入有误,跳转登录页面,提示:验证码错误
    * 如果全部输入正确,则跳转到主页 success.jsp,提示:${用户名}$,欢迎您

package cn.itcast.servlet;

import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.util.Random;

@WebServlet("/checkCodeServlet")
public class CheckCodeServlet extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 创建对象,在内存中代表图片
        int width = 100;
        int height = 50;
        BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);

        // 美化图片
        // 填充背景色
        Graphics g = image.getGraphics();
        g.setColor(Color.cyan);
        g.fillRect(0, 0, width, height);
        // 画边框
        g.setColor(Color.BLACK);
        g.drawRect(0, 0, width - 1, height - 1);

        String str = "ABCEEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwyz0123456789";
        // 生成随机角标
        Random ran = new Random();

        StringBuilder sb = new StringBuilder();
        for (int i = 1; i <= 4; i++) {
            int index = ran.nextInt(str.length());
            //获取字符
            char ch = str.charAt(index);
            sb.append(ch);
            // 写验证码
            g.drawString(ch + "", width / 5 * i, height / 2);

        }
        String checkCode_session = sb.toString();
        // 将验证码存入session
        req.getSession().setAttribute("checkCode_session", checkCode_session);
        // 画干扰线
        g.setColor(Color.green);
        // 随机生成坐标点
        for (int i = 0; i < 10; i++) {
            int x1 = ran.nextInt(width);
            int x2 = ran.nextInt(width);
            int y1 = ran.nextInt(height);
            int y2 = ran.nextInt(height);
            g.drawLine(x1, x2, y1, y2);
        }

        // 将图片输出到页面展示
        ImageIO.write(image, "jpg", resp.getOutputStream());
    }

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        this.doPost(req, resp);
    }
}
CheckCodeServlet
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>login</title>

    <style>
        div {
            color: red;
        }
    </style>
</head>
<body>
<form action="/day16/loginServlet" method="post">
    <table>
        <tr>
            <td>用户名</td>
            <td><input type="text" name="username"></td>
        </tr>
        <tr>
            <td>密码</td>
            <td><input type="password" name="password"></td>
        </tr>
        <tr>
            <td>验证码</td>
            <td><input type="text" name="checkCode"></td>
        </tr>
        <tr>
            <td colspan="2"><img src="/day16/checkCodeServlet" id="img"></td>
        </tr>
        <tr>
            <td colspan="2"><input type="submit" value="登录"></td>
        </tr>
    </table>
</form>

<div>
    <%=request.getAttribute("cc_error") == null ? "" : request.getAttribute("cc_error")%>
</div>
<div>
    <%=request.getAttribute("login_error") == null ? "" : request.getAttribute("login_error")%>
</div>


<script>
    window.onload = function (ev) {
        document.getElementById("img").onclick = function (ev1) {
            this.src = "/day16/checkCodeServlet?time=" + new Date().getTime();
        }
    };
</script>
</body>
</html>
login.jsp
package cn.itcast.servlet;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;

@WebServlet("/loginServlet")
public class loginServlet extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 设置request编码
        request.setCharacterEncoding("utf-8");

        // 获取参数Map
        String username = request.getParameter("username");
        String password = request.getParameter("password");
        String checkCode = request.getParameter("checkCode");

        // 获取生成的验证码
        HttpSession session = request.getSession();
        String checkCode_session = (String) session.getAttribute("checkCode_session");

        // 删除 session中存储的验证码
        session.removeAttribute("checkCode_session");

        // 判断验证码是否正确
        if (checkCode_session != null && checkCode_session.equalsIgnoreCase(checkCode)) {
            // 忽略大小写比较,验证码一致
            // 判断用户名和密码是否一致
            if ("zhangsan".equals(username) && "123".equals(password)) {
                // 登录成功
                // 存储信息
                session.setAttribute("user", username);
                // 重定向
                response.sendRedirect(request.getContextPath() + "/success.jsp");
            } else {
                // 登录失败
                request.setAttribute("login_error", "用户名或密码错误");
                request.getRequestDispatcher("/login.jsp").forward(request, response);

            }
        } else {
            // 验证码不一致
            // 存储提示信息到request
            request.setAttribute("cc_error", "验证码错误");
            request.getRequestDispatcher("/login.jsp").forward(request, response);
        }
    }

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request, response);
    }
}
LoginServlet
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<h1><%=request.getSession().getAttribute("user")%>,欢迎您</h1>
</body>
</html>
success.jsp
原文地址:https://www.cnblogs.com/zhaochuming/p/13590538.html