zoukankan      html  css  js  c++  java
  • 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
  • 相关阅读:
    K-Means原理及代码实现
    Windows 10安装Tomcat
    Maven笔记四:在Eclipse中创建java web项目
    Maven笔记三:使用Ecplise创建一个Maven项目
    Maven笔记二:Maven仓库
    Windows 10 安装Eclipse
    Maven笔记一:Maven安装与配置(Windows 10)
    Docker学习笔记五:Dockerfile
    Docker学习笔记五:docker简单实践(Docker部署Java应用)
    Zabbix笔记三:zabbix监控指标
  • 原文地址:https://www.cnblogs.com/zhaochuming/p/13590538.html
Copyright © 2011-2022 走看看