当前位置: 首页 > news >正文

Servlet开发-session和cookie理解案例-登录页面

项目展示

        进入登录页面,输入正确的用户名和密码以后会自动跳到主页

        登录成功以后打印用户名以及上次登录的时间,如果浏览器和客户端都保存有上次登录的信息,则不需要登录就可以进入主页        

编码思路

        1.首先提供一个登录的前端页面,用户进入登录页面,输入用户名和密码,根据用户名和密码发送一个post类型的HTTP请求给服务器

        2.服务器接收到HTTP请求以后,验证用户名和密码是否正确,如果正确,则在服务器中为该用户创建一个会话session,session中包含的键值对有用户的用户名和用户登录的时间,服务器把创建session产生的唯一的JSESSIONID通过HTTP响应发送给浏览器,浏览器将JSESSIONID保存到Cookie中(下次发送HTTP请求会携带着Cookie中的内容)

        3.会话创建成功以后,服务器会向浏览器发送一个重定向的HTTP响应,让浏览器发送一个访问主页的gei类型的HTTP请求,这样服务器就会向浏览器发送主页的HTTP响应,就达到了登录成功以后自动跳转到主页的效果

代码及其解析

        1.首先编写前端代码,让用户输入用户名和密码,发送一个POST类型的HTTP请求(携带着用户名和密码)给服务器

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>登录页面</title>
</head>
<body><!-- 通过from表单构造http请求 --><form action="login" method="post"><input type="text" name="username"><input type="password" name="password"><input type="submit" value="登录"></form>
</body>
</html>

        2.编写登录逻辑的后端代码,解析已经写在了代码中,推荐复制到idea中查看

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;/*** Created with IntelliJ IDEA.* Description:* User: wuyulin* Date: 2023-09-28* Time: 8:08*/
//登录
@WebServlet("/login")
public class LoginServlet extends HttpServlet {//客户会输入用户名和密码,通过post类型的HTTP请求发送给服务器@Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//获取用户名和密码//由于登录的HTTP请求是通过from表单构造发送的,所以是键值对结构的数据,通过getParameter方法便可以通过key值获得value值String username=req.getParameter("username");String password=req.getParameter("password");//检查用户名和密码是否符合要求if(username==null||password==null||username.equals("")||password.equals("")){//用户名和密码不符合要求,给用户返回HTTP响应做出提醒resp.setContentType("text/html; charset=utf8");resp.getWriter().write("用户名和密码不符合要求");return;}//验证用户名和密码是否正确//正常来说,客户的用户名和密码会保存到数据库中//验证用户名和密码会先在数据库中查找用户名是否存在,然后再检验密码是否正确//但现在简单实现的话,就约定用户名username=“zhangsan”,密码password=“123”//用户名错误if(!username.equals("zhangsan")){resp.setContentType("text/html; charset=utf8");resp.getWriter().write("用户名错误");return;}//密码错误if(!password.equals("123")){resp.setContentType("text/html; charset=utf8");resp.getWriter().write("密码错误");return;}//用户名和密码没有错误就登陆成功,可以给用户创建会话(session)//getSession背后做的事情://1.先读取req中的Cookie,看Cookie中是否有JSESSIONID属性,以及值是什么//  如果没有,就认为需要创建一个session并生成一个唯一的JSESSIONID//  如果有,就会拿着这个JSESSIONID去查询服务器中是否有对应的session存在//      要是session存在就会返回这个session//      要是session不存在就会创建一个session并生成一个唯一的JSESSIONID//2.当前是第一次进行登录操作的话HTTP请求的cookie中就肯定没有JSESSIONID,就会创建一个session并生成一个唯一的JSESSIONID//以JSESSIONID为key,session为value,把这个键值对插入到服务器存储session的哈希表中//3.刚才生成的JSESSIONID又会通过addCookie方法添加到HTTP响应中,此时HTTP响应就会带有Set-Cookie字段//这里的值就是JSESSIONID=xxxxxxx,通过HTTP响应传递浏览器,浏览器通过cookie保存这个JSESSIONID,之后发送的HTTP请求就会带着这个JSESSIONIDHttpSession session=req.getSession(true);//session中其实就是一些程序猿自定义的键值对//向session中传入一些键值对session.setAttribute("username",username);session.setAttribute("time",System.currentTimeMillis());//会话创建完毕,用户登录成功,自动从登录页面跳转到主页(发送重定向响应给客户端)//此处约定主页的路径是indexresp.sendRedirect("index");}
}

        3.主页的后端代码

        该案例中并没有专门为主页创建一个HTML页面,直接通过后端代码发送字符串表示即可

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;/*** Created with IntelliJ IDEA.* Description:* User: wuyulin* Date: 2023-09-28* Time: 10:19*///通过Servlet生成一个主页
@WebServlet("/index")
public class IndexServlet extends HttpServlet {//用户登录成功收到一个重定向的HTTP响应以后就会发送一个get类型的HTTP请求给服务器,请求获取index路径的http响应@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//验证用户的登录状态//当HTTP请求的Cookie中有JSESSIONID的话就根据JSESSIONID找到对应的session//要是没有JSESSIONID或者通过JSESSIONID找不到对应的session就直接返回nullHttpSession session=req.getSession(false);//用户没有登录if(session==null){//返回HTTP响应提醒用户登录resp.setContentType("text/html; charset=utf8");resp.getWriter().write("请先登录再来访问主页");return;}//用户已经登录成功//就可以取出session中的attribute//getAttribute方法可以通过key值获取到Attribute中的value值,但返回的类型是Object,所以要进行强转String username=(String) session.getAttribute("username");Long time=(Long) session.getAttribute("time");//获取到session中的username和time以后,根据相关内容构造主页即可resp.setContentType("text/html; charset=utf8");resp.getWriter().write("欢迎"+username+"! "+"上次登录时间为"+time);}
}

核心-session与cookie的交互过程

        1.首先,当用户是第一次登录时,发送给服务器的HTTP请求的中就没有Cookie(没有JSESSIONID),要是用户直接访问主页的话,服务器会先检查HTTP请求有没有Cookie属性,要是没有就会直接拒绝用户的访问,提醒用户先进行登录,要是有Cookie就会取出其中的JSESSIONID去浏览器中查找对应的session,要是没有找到的话也会直接拒绝用户的访问

        2.用户登录成功以后,服务器就会为用户创建一个对应的会话(session),并生成一个唯一的JSESSIONID,将JSESSIONID为key,session为value的键值对保存到服务器类似于哈希表的数据结构中,并将JSESSIONID通过HTTP响应发送给浏览器,浏览器就会保存这个JSESSIONID到Cookie中,下次发送HTTP请求的时候会带上Cookie中的内容

        3.此时要是用户退出网站,重新尝试获取主页信息的时候,发送的HTTP请求中就会带有Cookie,服务器收到HTTP请求以后就会检查Cookie中的JSESSIONID,通过JSESSIONID在服务器中找到对应的session,获得用户的相关信息(如本例中就获得了用户的用户名以及上次登录的时间),此时就不需要用户输入用户名和密码,直接通过HTTP响应返回主页的内容给浏览器

        以上的大部分操作在getSession方法中进行,可以看代码中的注释进行了解

相关文章:

Servlet开发-session和cookie理解案例-登录页面

项目展示 进入登录页面&#xff0c;输入正确的用户名和密码以后会自动跳到主页 登录成功以后打印用户名以及上次登录的时间&#xff0c;如果浏览器和客户端都保存有上次登录的信息&#xff0c;则不需要登录就可以进入主页 编码思路 1.首先提供一个登录的前端页面&…...

Polygon Miden:扩展以太坊功能集的ZK-optimized rollup

1. 引言 Polygon Miden定位为zkVM&#xff0c;定于2023年Q4上公开测试网。 zk、zkVM、zkEVM及其未来中指出&#xff0c;当前主要有3种类型的zkVM&#xff0c;括号内为其相应的指令集&#xff1a; mainstream&#xff08;WASM, RISC-V&#xff09;EVM&#xff08;EVM bytecod…...

[题]宝物筛选 #单调队列优化

五、宝物筛选&#xff08;洛谷P1776&#xff09; 题目链接 好家伙&#xff0c;找到了一个之前学习多重背包优化时的错误…… 之前记的笔记还是很有用的…… #include<bits/stdc.h> using namespace std; const int N 1e5 10; int f[N]; int n, m; int v, w, s; int l…...

.NET的键盘Hook管理类,用于禁用键盘输入和切换

一、MyHook帮助类 此类需要编写指定屏蔽的按键&#xff0c;灵活性差。 using System; using System.Runtime.InteropServices; using System.Diagnostics; using System.Windows.Forms; using Microsoft.Win32;namespace MyHookClass {/// <summary>/// 类一/// </su…...

Anaconda Jupyter

&#x1f64c;秋名山码民的主页 &#x1f602;oi退役选手&#xff0c;Java、大数据、单片机、IoT均有所涉猎&#xff0c;热爱技术&#xff0c;技术无罪 &#x1f389;欢迎关注&#x1f50e;点赞&#x1f44d;收藏⭐️留言&#x1f4dd; 获取源码&#xff0c;添加WX 目录 前言An…...

Unity中Shader的前向渲染路径ForwardRenderingPath

文章目录 前言一、前向渲染路径的特点二、渲染方式1、逐像素(效果最好)2、逐顶点(效果次之)3、SH球谐(效果最差) 三、Unity中对灯光设置 后&#xff0c;自动选择对应的渲染方式1、ForwardBase仅用于一个逐像素的平行灯&#xff0c;以及所有的逐顶点与SH2、ForwardAdd用于其他所…...

简历项目优化关键方法论-START

START方法论是非常著名的面试法则&#xff0c;经常被面试官使用的工具 Situation:情况、事情、项目需求是在什么情况下发生Task:任务&#xff0c;你负责的做的是什么Action:动作&#xff0c;针对这样的情况分析&#xff0c;你采用了什么行动方式Result:结果&#xff0c;在这样…...

TensorFlow学习1:使用官方模型进行图片分类

前言 人工智能以后会越来越发达&#xff0c;趁着现在简单学习一下。机器学习框架有很多&#xff0c;这里觉得学习谷歌的 TensorFlow&#xff0c;谷歌的技术还是很有保证的&#xff0c;另外TensorFlow 的中文文档真的很友好。 文档&#xff1a; https://tensorflow.google.cn/…...

C++ 并发编程实战 第八章 设计并发代码 一

目录 8.1 在线程间切分任务 8.1.1 先在线程间切分数据&#xff0c;再开始处理 8.1.2 以递归方式划分数据 8.1.3 依据工作类别划分任务 借多线程分离关注点需防范两大风险 在线程间按流程划分任务 8.2 影响并发性能的因素 8.2.1 处理器的数量 8.2.2 数据竞争和缓存兵乓…...

设计模式8、装饰者模式 Decorator

解释说明&#xff1a;动态地给一个对象增加一些额外的职责。就扩展功能而言&#xff0c;装饰模式提供了一种比使用子类更加灵活的替代方案 抽象构件&#xff08;Component&#xff09;&#xff1a;定义一个抽象接口以规范准备收附加责任的对象 具体构件&#xff08;ConcreteCom…...

抖音开放平台第三方代小程序开发,一整套流程

大家好&#xff0c;我是小悟 抖音小程序第三方平台开发着力于解决抖音生态体系内的小程序管理问题&#xff0c;一套模板&#xff0c;随处部署。能尽可能地减少服务商的开发成本&#xff0c;服务商只用开发一套小程序代码作为模板就可以快速批量的孵化出大量的商家小程序。 第…...

Flutter笔记:滚动之-无限滚动与动态加载的实现(GetX简单状态管理版)

Flutter笔记 无限滚动与动态加载的实现&#xff08;GeX简单状态管理版&#xff09; 作者&#xff1a;李俊才 &#xff08;jcLee95&#xff09;&#xff1a;https://blog.csdn.net/qq_28550263 邮箱 &#xff1a;291148484163.com 本文地址&#xff1a;https://blog.csdn.net/qq…...

前端架构师之02_ES6_高级

1 类和继承 1.1 class类 JavaScript 语言中&#xff0c;生成实例对象的传统方法是通过构造函数。 // ES5 创建对象 // 创建一个类&#xff0c;用户名 密码 function User(name,pass){// 添加属性this.name name;this.pass pass; } // 用 原型 添加方法 User.prototype.sho…...

VScode多文件编译/调试配置

之前都是在Visual Studio写C/C&#xff0c;最近想换到VScode&#xff0c;折腾半天把launch.json和tasks.json配好了&#xff08;虽然不懂为什么&#xff0c;但确实能用了&#xff09;&#xff0c;在此做个记录。 参考资料&#xff1a;1&#xff0c;2&#xff0c;3 环境&#…...

K折交叉验证——cross_val_score函数使用说明

在机器学习中&#xff0c;许多算法中多个超参数&#xff0c;超参数的取值不同会导致结果差异很大&#xff0c;如何确定最优的超参数&#xff1f;此时就需要进行交叉验证的方法&#xff0c;sklearn给我们提供了相应的cross_val_score函数&#xff0c;可对数据集进行交叉验证划分…...

2023.09.30使用golang1.18编译Hel10-Web/Databasetools的windows版

#Go 1.21新增的 log/slog 完美解决了以上问题&#xff0c;并且带来了很多其他很实用的特性。 本次编译不使用log/slog 包 su - echo $GOPATH ;echo $GOROOT; cd /tmp; busybox wget --no-check-certificate https://go.dev/dl/go1.18.linux-amd64.tar.gz;\ which tar&&am…...

React简介

react作为前端主流框架之一&#xff0c;因其语法接近原生JavaScript语法而广受欢迎。其生态丰富&#xff0c;常用的就有react-router、react-redux等插件&#xff0c;还有与其匹配的UI组件库antd。而且其还有用于移动端开发的react-native库&#xff0c;因此&#xff0c;react值…...

链表经典面试题(一)

面试题 1.反转链表的题目2.反转链表的图文分析3.反转链表的代码实现 1.反转链表的题目 2.反转链表的图文分析 我们在实现反转链表的时候,是将后面的元素变前面&#xff0c;前面的元素变后面&#xff0c;那么我们是否可以理解为&#xff0c;用头插法的思想来完成反转链表呢&…...

体验亚马逊的 CodeWhisperer 感觉

CodeWhisperer 是亚马逊推出的辅助编程工具&#xff0c;在程序员写代码时&#xff0c;它能根据其内容生成多种代码建议。 CodeWhisperer 目前已支持近10几种语言&#xff0c;我是用 java 语言&#xff0c;用的开发工具是 idea&#xff0c;说一下我用的情况。 亚马逊云科技开发…...

6、行内元素和块元素

6、行内元素和块元素 一、块元素 无论内容多少&#xff0c;该元素独占一行 如p标签、标题标签&#xff08;h1-h6…&#xff09; 二、行内元素 内容撑开宽度、左右都是行内元素的可以排在一行 一些元素如果能够摆放在一行都可以用行内元素&#xff0c;但是如果需要换行就需…...

在软件开发中正确使用MySQL日期时间类型的深度解析

在日常软件开发场景中&#xff0c;时间信息的存储是底层且核心的需求。从金融交易的精确记账时间、用户操作的行为日志&#xff0c;到供应链系统的物流节点时间戳&#xff0c;时间数据的准确性直接决定业务逻辑的可靠性。MySQL作为主流关系型数据库&#xff0c;其日期时间类型的…...

FFmpeg 低延迟同屏方案

引言 在实时互动需求激增的当下&#xff0c;无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作&#xff0c;还是游戏直播的画面实时传输&#xff0c;低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架&#xff0c;凭借其灵活的编解码、数据…...

YSYX学习记录(八)

C语言&#xff0c;练习0&#xff1a; 先创建一个文件夹&#xff0c;我用的是物理机&#xff1a; 安装build-essential 练习1&#xff1a; 我注释掉了 #include <stdio.h> 出现下面错误 在你的文本编辑器中打开ex1文件&#xff0c;随机修改或删除一部分&#xff0c;之后…...

蓝桥杯 2024 15届国赛 A组 儿童节快乐

P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡&#xff0c;轻快的音乐在耳边持续回荡&#xff0c;小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下&#xff0c;六一来了。 今天是六一儿童节&#xff0c;小蓝老师为了让大家在节…...

HTML 列表、表格、表单

1 列表标签 作用&#xff1a;布局内容排列整齐的区域 列表分类&#xff1a;无序列表、有序列表、定义列表。 例如&#xff1a; 1.1 无序列表 标签&#xff1a;ul 嵌套 li&#xff0c;ul是无序列表&#xff0c;li是列表条目。 注意事项&#xff1a; ul 标签里面只能包裹 li…...

江苏艾立泰跨国资源接力:废料变黄金的绿色供应链革命

在华东塑料包装行业面临限塑令深度调整的背景下&#xff0c;江苏艾立泰以一场跨国资源接力的创新实践&#xff0c;重新定义了绿色供应链的边界。 跨国回收网络&#xff1a;废料变黄金的全球棋局 艾立泰在欧洲、东南亚建立再生塑料回收点&#xff0c;将海外废弃包装箱通过标准…...

【HTML-16】深入理解HTML中的块元素与行内元素

HTML元素根据其显示特性可以分为两大类&#xff1a;块元素(Block-level Elements)和行内元素(Inline Elements)。理解这两者的区别对于构建良好的网页布局至关重要。本文将全面解析这两种元素的特性、区别以及实际应用场景。 1. 块元素(Block-level Elements) 1.1 基本特性 …...

[Java恶补day16] 238.除自身以外数组的乘积

给你一个整数数组 nums&#xff0c;返回 数组 answer &#xff0c;其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&#xff0c;且在 O(n) 时间复杂度…...

均衡后的SNRSINR

本文主要摘自参考文献中的前两篇&#xff0c;相关文献中经常会出现MIMO检测后的SINR不过一直没有找到相关数学推到过程&#xff0c;其中文献[1]中给出了相关原理在此仅做记录。 1. 系统模型 复信道模型 n t n_t nt​ 根发送天线&#xff0c; n r n_r nr​ 根接收天线的 MIMO 系…...

MyBatis中关于缓存的理解

MyBatis缓存 MyBatis系统当中默认定义两级缓存&#xff1a;一级缓存、二级缓存 默认情况下&#xff0c;只有一级缓存开启&#xff08;sqlSession级别的缓存&#xff09;二级缓存需要手动开启配置&#xff0c;需要局域namespace级别的缓存 一级缓存&#xff08;本地缓存&#…...