JavaWeb-Session

1. Session简单介绍

在WEB开发中,服务器可以为每个浏览器用户创建一个会话对象 ( session对象),注意:一个浏览器用户独占一个session对象(默认情况下)。 因此,在需要保存用户数据时,服务器程序可以把用户数据写到浏览器用户独占的session中,当用户使用浏览器访问该web应用的其它servlet时,其它servlet可以从用户的session中取出该用户的数据,为用户服务,从而实现数据在多个页面中的共享。

2. Session和Cookie的主要区别

  • Cookie是把用户的数据写到用户的浏览器。
  • Session技术把用户的数据写到用户独占的session中,tomcat服务器内存中。
  • Session对象由服务器(Tomcat)创建,开发人员可以调用request对象的getSession()方法得到session对象

3. session机制演示图

image.png
image.png

服务器通过request对象的getSession方法创建出session对象后,会把session的id号,以cookie的形式回写给客户机,这样,只要客户机的浏览器不关,再去访问服务器时,都会带着session的id号去,服务器发现客户机浏览器带session id过来了,就会使用内存中与之对应的session为之服务。

前台页面

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<a href="session.do">第一次访问session.do</a>
<a href="sessiontest.do">第二次访问sessiontest.do</a>
</body>
</html>

第一次访问

package com.lty.servlet;

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

@WebServlet(name = "SessionServlet" ,urlPatterns = "/session.do")
public class SessionServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 每一个session对象都存在着唯一的ID
        HttpSession httpSession = request.getSession();
        httpSession.setAttribute("user","zhangsan");
        System.out.println(httpSession.getId());

        // Tomcat服务器默默做的一件事
//        Cookie cookie = new Cookie("JSESSIONID",httpSession.getId());
//        response.addCookie(cookie);
    }
}

第二次访问

package com.lty.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(name = "SessionTestServlet",urlPatterns = "/sessiontest.do")
public class SessionTestServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 获取第一次服务器创建给用户的Session
        HttpSession httpSession = request.getSession();
        String username = (String)httpSession.getAttribute("user");
        System.out.println(username);
        System.out.println( httpSession.getId());
    }
}
image.png

4. Session cookie

session通过SessionID来区分不同的客户,session是以cookie为基础的,系统会创造一个名为JSESSIONID的输出cookie ,这称之为session cookie,以区别persistent cookies(也就是我们通常所说cookie),session cookie是存储于浏览器内存中的,并不是写到硬盘上的, session cookie针对某一次会话而言,会话结束session cookie也就随着消失了,而persistent cookie只是存在于客户端硬盘上的一段文本。

关闭浏览器,只会使浏览器端内存里的session cookie消失,但不会使保存在服务器端的session对象消失,同样也不会使已经保存到硬盘上的持久化cookie消失。

持久化Session cookie
关闭浏览器,一小时之内打开浏览器,session仍然有效(通过使用cookie类的setMaxAge(秒)方法设置持久化的时效)

package com.lty.servlet;

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

@WebServlet(name = "TestServlet",urlPatterns = "/test")
public class TestServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        HttpSession httpSession = request.getSession();
        Cookie cookie = new Cookie("JSESSIONID",httpSession.getId());
        cookie.setMaxAge(60*60);// 关闭浏览器,一小时之内打开浏览器,session仍然有效
    }
}

5. HttpSession的生命周期

问题:用户开一个浏览器访问一个网站,服务器是不是针对这次会话创建一个session?
答:不是的。session的创建时机是在程序中第一次去执行request.getSession();这个代码,服务器才会为你创建session。
问题:关闭浏览器,会话结束,session是不是就销毁了呢?
答:不是的。session是30分钟(默认30分钟)没人用了才会死,服务器会自动摧毁session。

session何时会销毁
(1)调用invalidate()方法,该方法使HttpSession立即失效

package com.lty.servlet;

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

@WebServlet(name = "TestServlet",urlPatterns = "/test")
public class TestServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        HttpSession httpSession = request.getSession();
        httpSession.invalidate();
    }
}

(2)服务器卸载了当前WEB应用
(3)超出session过期时间

package com.lty.servlet;

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

@WebServlet(name = "TestServlet",urlPatterns = "/test")
public class TestServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        HttpSession httpSession = request.getSession();
        httpSession.setMaxInactiveInterval(10);// 单位: 秒
    }
}

web.xml文件中也可以修改session的过期时间(全局的,所有的session的过期时间,若想单独设置还是需要像上面那样进行单独session设置),单位:分钟

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0">
    <session-config>
        <session-timeout>30</session-timeout>
    </session-config>
</web-app>

6. session小练习(利用session完成用户登录)

java实体类对象 User.java

package com.lty.bean;

public class User {
    private String username;
    private String password;

    public User() {
    }

    public User(String username, String password) {
        this.username = username;
        this.password = password;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }
}

DB java类(利用list集合来模拟向数据库中查询用户信息的操作过程)

package com.lty.bean;

import java.util.ArrayList;
import java.util.List;

public class DB {
    public static List<User> list = new ArrayList<User>();

    static{
        list.add(new User("aaa","123"));
        list.add(new User("bbb","123"));
        list.add(new User("ccc","123"));
    }
    public static List<User> getAll(){
        return list;
    }
}

主页面index.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
  <head>
    <title>$Title$</title>
  </head>
  <body>
  欢迎你:${user.username}
  <a href="login.jsp">登录</a>
  <a href="LogoutServlet">退出登录</a>
  <br>

  <a href="SessionDemo1">购买</a>
  <br>
  </body>
</html>

登录页面 login.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<form action="LoginServlet" method="post">
    用户名:<input type="text" name="username">
    密码:<input type="password" name="password">
    <input type="submit" value="登录">
</form>
</body>
</html>

购买页面 SessionDemo1.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
欢迎你:${user.username}
</body>
</html>

后台servlet

package com.lty.servlet;

import com.lty.bean.DB;
import com.lty.bean.User;

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.io.IOException;
import java.io.PrintWriter;
import java.util.List;

@WebServlet(name = "LoginServlet" , urlPatterns = "/LoginServlet")
public class LoginServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        response.setContentType("text/html;charset=utf-8");

        PrintWriter out = response.getWriter();
        String username = request.getParameter("username");
        String password = request.getParameter("password");

        List<User> list = DB.getAll();
        for(User user : list){
            // 用户登录成功
            if(user.getUsername().equals(username) && user.getPassword().equals(password)){
                request.getSession().setAttribute("user",user);
                response.sendRedirect("index.jsp");
                return;
            }
        }

        out.print("用户名或密码不正确!!!");
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }
}
package com.lty.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 java.io.IOException;

@WebServlet(name = "LogoutServlet" , urlPatterns = "/LogoutServlet")
public class LogoutServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 立即彻底销毁当前session对象
        request.getSession().invalidate();

        response.sendRedirect("index.jsp");
    }
}
package com.lty.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 java.io.IOException;

@WebServlet(name = "SessionDemo1" , urlPatterns = "/SessionDemo1")
public class SessionDemo1 extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        request.getRequestDispatcher("SessionDemo1.jsp").forward(request,response);
    }
}

7. 使用Session实现验证码

Session的典型案例:一次性验证码
java工具类 GenerateCode.java(将随机生成的由四个字符组成的字符串用图片进行渲染)

package com.lty.util;

import java.awt.*;
import java.awt.image.BufferedImage;
import java.util.Random;

public class GenerateCode {

    // 设置验证码图片的宽度,高度,验证码的个数
    private int width = 152;
    private int height = 40;

    // 验证码字体的高度
    private int fontHeight = 38;

    // 验证码中的单个字符基线。即:验证码中的单个字符位于验证码图形左上角的(codeX,codeY)位置处
    private int codeX = 25;
    private int codeY = 36;

    public BufferedImage Generate(String code){

        // 定义一个类型为 BufferedImage.TYPE_INT_BGR 类型的图像缓存
        BufferedImage buffImg = null;
        buffImg = new BufferedImage(width,height,BufferedImage.TYPE_3BYTE_BGR);

        // 在 buffImg 中创建一个 Graphics2D 图像
        Graphics2D graphics = null;
        graphics = buffImg.createGraphics();

        // 设置一个颜色,使 Graphics2D 对象的后续图形使用这个颜色
        graphics.setColor(Color.WHITE);

        // 填充一个指定的矩形:x - 要填充矩形的 x 坐标; y - 要填充矩形的 y 坐标; width - 要填充矩形的宽度; height - 要填充矩形的高度
        graphics.fillRect(0,0,width,height);

        // 创建一个 Font 对象:name - 字体名称; style - Font 的样式常量; size - Font 的点大小
        Font font = null;
        font = new Font("",Font.BOLD,fontHeight);
        // 使 Graphics2D 对象的后续图形使用此字体
        graphics.setFont(font);
        graphics.setColor(Color.BLACK);

        // 绘制指定矩形的边框,绘制出的矩形将比构件宽一个也高一个像素
        graphics.drawRect(0,0,width - 1,height - 1);

        // 随机产生 15 条干扰线,使图像中的验证码不易被其它程序探测到
        Random random = null;
        random = new Random();
        graphics.setColor(Color.GREEN);
        for(int i = 0; i < 30; i++){
            int x = random.nextInt(width);
            int y = random.nextInt(height);
            int x1 = random.nextInt(20);
            int y1 = random.nextInt(20);
            graphics.drawLine(x,y,x + x1,y + y1);
        }

        graphics.setColor(Color.BLACK);
        for(int i = 0;i < code.length();i++){

            // 用随机产生的颜色将验证码绘制到图像中
            graphics.drawString(String.valueOf(code.charAt(i)),(i + 1)* codeX,codeY);
        }

        return buffImg;
    }
}

测试生成验证码图片的java类(将生成的验证码图片保存到电脑磁盘中)

package com.lty.util;

import javax.imageio.ImageIO;
import java.awt.image.BufferedImage;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

public class Main {
    public static void main(String[] args) throws IOException {
        GenerateCode generateCode = new GenerateCode();
        BufferedImage bufferedImage = generateCode.Generate("lisi");

        // 把图片保存到磁盘
        FileOutputStream fileOutputStream = new FileOutputStream("d:/abc.jpg");
        ImageIO.write(bufferedImage,"jpeg",fileOutputStream);
        System.out.println("成功");

    }
}

登录页面 login.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
    <script>
        window.onload = function () {
            var img = document.getElementById('code');
            img.onclick = function () {
                var d = new Date();
                // alert(d.getTime());
                // 为了让浏览器感觉到每一次请求的地址不一样(不会使用之前缓存的图片),每一次都会发出新的请求,使服务器生成新的验证码
                img.src = "ValidateColorServlet?time=" + d.getTime();
            }
        }
    </script>
</head>
<body>
<form action="LoginServlet" method="post">
    username:<input type="text" name="username">
    验证码:<input type="text" name="check">
    <img src="ValidateColorServlet" id="code">
    <input type="submit" value="登录">
</form>
</body>
</html>

后台servlet 随机生成四位验证码

package com.lty.servlet;

import com.lty.util.GenerateCode;

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

@WebServlet(name = "ValidateColorServlet" , urlPatterns = "/ValidateColorServlet")
public class ValidateColorServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//        System.out.println("ValidateColorServlet");
        //回应一张图片的全部字节 --- 图片由一堆字节构成 浏览器进行解析 显示成图片
        //这张图片是一个验证码图片
        //1.在内存中生成这张图片
        //1.1图片上的文字如何生成?
        //随机生成一个4个字符的字符串(a-z,0-9组成)
        char [] codeSequence = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz123456789".toCharArray();
        String code = "";
        for(int i = 0; i < 4; i++){
            Random random = new Random();
            int index = random.nextInt(codeSequence.length);
            char c = codeSequence[index];
            code += c;
        }

        System.out.println(code);

        request.getSession().setAttribute("code",code);

        GenerateCode generateCode = new GenerateCode();
        BufferedImage bufferedImage = generateCode.Generate(code);

        // 向客户端浏览器进行输出
        ServletOutputStream outputStream = response.getOutputStream();
        ImageIO.write(bufferedImage,"jpeg",outputStream);
    }
}

校验用户输入的验证码与随机生成的验证码的一致性

package com.lty.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 java.io.IOException;

@WebServlet(name = "LoginServlet" , urlPatterns = "/LoginServlet")
public class LoginServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        response.setContentType("text/html;charset=utf-8");
        String checkcode = request.getParameter("check");// 用户输入的验证码
        String genCode = (String) request.getSession().getAttribute("code");// 当时随机生成的验证码
        if(checkcode.equalsIgnoreCase(genCode)){
            response.getWriter().println("验证码正确");
        }else{
            response.getWriter().println("验证码不正确");
        }

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 210,914评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 89,935评论 2 383
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,531评论 0 345
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,309评论 1 282
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,381评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,730评论 1 289
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,882评论 3 404
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,643评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,095评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,448评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,566评论 1 339
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,253评论 4 328
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,829评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,715评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,945评论 1 264
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,248评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,440评论 2 348