SpringBoot通过token实现用户互踢功能

认识token

所谓token,既用户能够在一定时间内证明自己身份的一长串字符串。正常的使用流程为:用户第一次登入——》服务器为该用户签发一份token——》进行其他服务请求时携带上token——》服务器判断此token在有效期内——》放行此次请求。

在上述过程中,用户只有在请求特定的接口时可以不用携带token,例如登入、请求一些基本的公共信息等。

通过token实现用户互踢

通过上述我们知道,用户在请求一些接口时需要用到token进行校验。那么要想通过token实现用户互踢的功能,其实就变得简单了。具体思路为:

①:设立一份token白名单

②:同一个账号多次登入时,新登入的用户将之前登入的用户token挤出白名单

这里需要注意的是:token无法主动设置某个token为无效状态。这也就意味着,我们需要设置一份白名单或者黑名单。

白名单:只有在白名单内的token才算是有效的。

黑名单:在黑名单内的token都是无效的。

具体实现

这里我使用的是白名单的方法,之所以使用白名单,是因为使用白名单所占用的空间小,因为一个用户正在有效的token只会有一个,而其无效的token可能会有多个。具体步骤如下:

1、token的实现

package org.example.untils;import com.auth0.jwt.JWT;
import com.auth0.jwt.JWTVerifier;
import com.auth0.jwt.algorithms.Algorithm;
import com.auth0.jwt.exceptions.JWTDecodeException;
import com.auth0.jwt.exceptions.JWTVerificationException;import java.util.Date;
import java.util.HashMap;
import java.util.Map;public class Token {/*** 过期30分钟* */private static final long EXPIRE_TIME = 30 * 60 * 1000;/*** jwt密钥* */private static final String SECRET = "jwt_secret";public static Map<String,String> map=new HashMap<>();//存放token的map集合public static Map<String, String> getMap() {return map;}public static void setMap(Map<String, String> map) {Token.map = map;}/*** 生成jwt字符串,30分钟后过期  JWT(json web token)* @param account* @return* */public static String sign(String account) {//签发tokentry {Date date = new Date(System.currentTimeMillis() + EXPIRE_TIME);Algorithm algorithm = Algorithm.HMAC256(SECRET);return JWT.create()//将account保存到token里面.withAudience(account)//五分钟后token过期.withExpiresAt(date)//token的密钥.sign(algorithm);} catch (Exception e) {e.printStackTrace();return null;}}/*** 根据token获取account* @param token* @return* */public static String getAccount(String token) {try {String account = JWT.decode(token).getAudience().get(0);return account;}catch (JWTDecodeException e) {return null;}}/*** 校验token* @param token* @return* */public static boolean checkSign(String token) {try {Algorithm algorithm  = Algorithm.HMAC256(SECRET);JWTVerifier verifier = JWT.require(algorithm)//.withClaim("username, username).build();verifier.verify(token);return true;} catch (JWTVerificationException e) {System.out.println("token无效");String account=Token.getAccount(token);//将该token从白名单内移除if(account!=null){Token.map.remove(account);//移出白名单}return false;//token无效}}
}

上述为token的实现,其中包括了token的签发,验证以及根据token获取账号。

2、用户登入时的方法实现

@GetMapping("/login")//用户登入public Result login(@Param("account") String account,@Param("password") String password){User user = userServiceImpl.login(account);if (user==null){return new Result(201,"账号不存在",null);} else if(user.getPassword().equals(password)){//密码正确Token.map.remove(account);//移除之前的tokenString token=Token.sign(account);//签发新的tokenToken.map.put(account,token);//将新的token移入有效tokenuser.setPassword("");return new Result(200,token,user);} else {return new Result(201,"账号/密码错误",null);}}

从上述代码可见,当登入成功时,会将上一次登入时留下的token移除白名单,并将最近登入生成的token放入白名单。

3、通过拦截器进行校验

package org.example.untils;import org.springframework.web.servlet.HandlerInterceptor;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.Objects;public class Interceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)throws Exception {String token=request.getHeader("token");String account=Token.getAccount(token);//通过token获取用户账号String usingToken=Token.map.get(account);//获取该用户最新的有效tokenif(!Objects.equals(token, usingToken)){//该token已经失效response.setStatus(401);return false;}//检查tokenif(Token.checkSign(token)){return true;} else {response.setStatus(401);return false;}}
}

在这里,我们会判断用户携带的token是否存在于白名单内,不存在则说明这次携带的token是无效的。

4、对拦截器进行注册并放行登入等接口

package org.example.untils;import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;@Configuration//定义此类为配置类
public class InterceptorConfig implements WebMvcConfigurer {@Overridepublic void addInterceptors(InterceptorRegistry registry) {//创建用户拦截器对象并指定其拦截的路径和排除的路径registry.addInterceptor(new Interceptor()).addPathPatterns("/**").excludePathPatterns("/user/login","/admin/login","/service/getAllServices","/shop/getShopById","/img/**");}
}

上述代码为放行了登入等接口。

总结

当然,在此次我的白名单使用的是Map存储的。网络上也有使用redis的好像,因为我目前并没有学习redis,大家感兴趣的可以试试使用redis。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/802063.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Pytorch中nn.Linear使用方法

nn.Linear定义一个神经网络的线性层&#xff1a; torch.nn.Linear(in_features, # 输入的神经元个数out_features, # 输出神经元个数biasTrue # 是否包含偏置)nn.Linear其实就是对输入&#xff08;n表示样本数量&#xff0c;i表示样本特…

【数据结构与算法】力扣 142. 环形链表 II

题目描述 给定一个链表的头节点 head &#xff0c;返回链表开始入环的第一个节点。 如果链表无环&#xff0c;则返回 null。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统…

华为海思校园招聘-芯片-数字 IC 方向 题目分享——第二套

华为海思校园招聘-芯片-数字 IC 方向 题目分享&#xff08;有参考答案&#xff09;——第二套&#xff08;共九套&#xff0c;每套四十个选择题&#xff09; 部分题目分享&#xff0c;完整版获取&#xff08;WX:didadidadidida313&#xff0c;加我备注&#xff1a;CSDN huawei…

Git-LFS 远程命令执行漏洞 CVE-2020-27955 漏洞复现

今天遇到了一个比较有意思的洞&#xff0c;复现一下下.......... 漏洞描述 Git LFS 是 Github 开发的一个 Git 的扩展&#xff0c;用于实现 Git 对大文件的支持 一些受影响的产品包括Git&#xff0c;GitHub CLI&#xff0c;GitHub Desktop&#xff0c;Visual Studio&#xff0…

51单片机之自己配串口寄存器实现波特率9600

本配置是根据手册进行开发配置的 1、首先配置SCON 所以综上所诉 SCON 0x40 &#xff08;0100 0000&#xff09; 2、PCON不用配置 3、配置定时器1 4、波特率的计算 5、配置AUXR 6、对比 7、实现 8、优化&#xff08;实现字符串&#xff09; 引入TI &#xff08;智能延时&…

对于嵌入式工程师,需要掌握的知识是广还是精?

我刚开始接触嵌入式的时候&#xff0c;感觉学这个好变态啊。 要学的东西太多了&#xff0c;数字电路、模拟电路、C语言、汇编、51单片机、Protel 99SE、Pcb Layout、STM32单片机、RTOS、Linux、ARM等等.... 可以说&#xff0c;随便拿个魔法电路出来&#xff0c;想达到精的程度&…

朋友圈文案示例

对于一张留着深思表情的人物照片&#xff1a;‘眼中的光芒胜过千言万语&#xff0c;沉思的瞬间&#xff0c;抚摩心灵的深处。 对于一张微笑着的老人肖像&#xff1a;‘满布皱纹的笑容&#xff0c;记录着岁月的沉淀和生活的甘甜。 对于一张孩子好奇探索的照片&#xff1a;‘童真…

【C++】C++11可变参数模板

&#x1f440;樊梓慕&#xff1a;个人主页 &#x1f3a5;个人专栏&#xff1a;《C语言》《数据结构》《蓝桥杯试题》《LeetCode刷题笔记》《实训项目》《C》《Linux》《算法》 &#x1f31d;每一个不曾起舞的日子&#xff0c;都是对生命的辜负 目录 前言 可变参数模板的定义…

算法训练营第35天|LeetCode 860.柠檬水找零 406.根据身高重建队列 452.用最少数量的箭引爆气球

LeetCode 860.柠檬水找零 题目链接&#xff1a; LeetCode 860.柠檬水找零 解题思路&#xff1a; 枚举法&#xff0c;写出所有找零的情况。 代码: class Solution { public:bool lemonadeChange(vector<int>& bills) {if(bills[0]!5) return false;unordered_ma…

Java绘图坐标体系

一、介绍 下图说明了Java坐标系。坐标原点位于左上角&#xff0c;以像素为单位。在Java坐标系中&#xff0c;第一个是x坐标&#xff0c;表示当前位置为水平方向&#xff0c;距离坐标原点x个像素&#xff1b;第二个是y坐标&#xff0c;表示当前位置为垂直方向&#xff0c;距离坐…

git lfs使用(huggingface下载大模型文件)

0、git lfs是Git的一个扩展工具,用于管理大型文件(如音视频文件、数据集等),避免将它们存储在Git仓库中,而是单独存储在Git LFS服务器上。 1. 安装 Linux安装步骤&#xff1a; curl -s https://packagecloud.io/install/repositories/github/git-lfs/script.deb.sh | sudo b…

Qt | .pro开发经验笔记

Qt | .pro开发经验笔记 【1】 Qt中 += 和 *= 的区别【2】构建识别平台【3】构建定位版本【4】构建引入第三方库【5】QCustomplot绘图性能的改善【6】Qt4/Qt5/Qt6版本控制【7】pro文件中写上版本号、程序图标、产品名称、版权所有、文件说明等信息(Qt5才支持)【8】管理员运行程…

matlab 直方图及分布拟合

datanormrnd(0,5,[1,500]); %频数图 histogram(data,30); %频率图 histogram(data,30,Normalization,probability);%分布拟合方法一 histfit(data,30); %直方图&正态分布拟合 xlim([-16,16]); %x范围 %单独画拟合曲线&#xff08;based on&#xff1a;频数直方图&#xff0…

LLM大语言模型(九):LangChain封装自定义的LLM

背景 想基于ChatGLM3-6B用LangChain做LLM应用&#xff0c;需要先了解下LangChain中对LLM的封装。本文以一个hello world的封装来示例。 LangChain中对LLM的封装 继承关系&#xff1a;BaseLanguageModel——》BaseLLM——》LLM LLM类 简化和LLM的交互 _call抽象方法定义 ab…

SpringBoot和SpringCloud,SpringCloudAlibaba版本依赖关系

SpringBoot和SpringCloud,SpringCloudAlibaba版本依赖关系 版本说明 原始文档 毕业版本依赖关系(推荐使用) 由于 Spring Boot 3.0&#xff0c;Spring Boot 2.7~2.4 和 2.4 以下版本之间变化较大&#xff0c;目前企业级客户老项目相关 Spring Boot 版本仍停留在 Spring Boot…

操作系统理论知识快速总览

操作系统整体架构 搬出考研时的思维导图 操作系统主要分为 批处理系统(老古董&#xff0c;基本不用了)实时操作系统(嵌入式中使用较多&#xff0c;RTOS)分时操作系统(PC中使用较多&#xff0c;Linux&#xff0c;Windows) 分时操作系统和实时操作系统的使用场景不同&#xf…

【蓝桥杯第十二届省赛B】(部分详解)

空间 8位1b 1kb1024b(2^10) 1mb1024kb(2^20) 时间显示 #include <iostream> using LLlong long; using namespace std; int main() {LL t;cin>>t;int HH,MM,SS;t/1000;SSt%60;//like370000ms370s,最后360转成分余下10st/60;MMt%60;t/60;HHt%24;printf("%02d:…

前端qiankun如何实现微应用路由跳转

appContext&#xff1a;qiankun 沙箱的上下文对象&#xff0c;包含了子应用的信息和一些常用的方法和属性config&#xff1a;子应用的配置对象globalProperties&#xff1a;子应用的全局属性对象&#xff0c;它可以被子应用中的所有组件和模块访问$mainRouter&#xff1a;这是父…

[C语言]——动态内存管理

目录 一.为什么要有动态内存分配 二.malloc和free 1.malloc 2.free 三.calloc和realloc 1.calloc 2.realloc 3.空间的释放​编辑 四.常见的动态内存的错误 1.对NULL指针的解引用操作 2.对动态开辟空间的越界访问 3.对非动态开辟内存使用free释放 4.使用free释放⼀块…

外汇110:谷歌起诉应用程序开发商伪造加密投资APP诈骗!

谷歌&#xff08;Google&#xff09;已对两家应用程序开发商提起诉讼&#xff0c;指控其参与“国际在线消费者投资欺诈计划”。该计划欺骗用户从 Google Play 商店和其他渠道下载虚假的安卓&#xff08;Android&#xff09;应用程序&#xff0c;并以承诺更高回报为幌子窃取他们…