首页 > 数据库 >token+redis使用

token+redis使用

时间:2024-10-29 20:47:44浏览次数:8  
标签:return String redis JWT token Token static 使用

在现代Web应用中,Token与Redis的结合使用已成为一种常见的身份验证和授权机制。

Token的作用:会话跟踪,每次客户端携带token访问服务器,服务器端校验token的正确性。        

好处:

  1. 灵活的过期策略:Redis可以设置Token的过期时间,一旦过期,系统需要重新验证用户身份。通过合理地设置过期时间,可以提高系统的安全性和性能。

  2. 安全性:Token通常包含敏感信息,如用户ID、权限等。将这些信息存储在Redis中,并采取适当的安全措施(如加密传输、访问控制等),可以保护Token不被未授权访问或篡改

JWT口令(token)

JSON Web Token(JWT)是为了在网络应用环境间传递声明而执行的一种基于JSON的开放标准((RFC 7519)。

JWT由三部分组成,分别是头部(Header)、负载(Payload)、签名(Signature),中间以(.)分隔

其中Healder是现阶段使用过的

Header中存储了所使用的加密算法和Token类型,例如:
{
  "alg": "HS256",
  "typ": "JWT"
}

若想会用token,先在pom.xml中导入依赖

<dependency>
    <groupId>com.auth0</groupId>
    <artifactId>java-jwt</artifactId>
    <version>3.2.0</version>
</dependency>

之后,我们要创建token工具类

@Component

public class TokenUtil {

    private static final Logger logger = LoggerFactory.getLogger(TokenUtil.class);
    /**
     * 过期时间
     */

    private static String accessTokenExpireTime;
    /**
     * JWT认证加密私钥(Base64加密)
     */
    private static String encryptJWTKey;

    /**
     * 解决@Value不能修饰static的问题
     */
    @Value("${accessTokenExpireTime}")
    public void setAccessTokenExpireTime(String accessTokenExpireTime) {
        TokenUtil.accessTokenExpireTime = accessTokenExpireTime;
    }

    @Value("${encryptJWTKey}")
    public void setEncryptJWTKey(String encryptJWTKey) {

        TokenUtil.encryptJWTKey = encryptJWTKey;
    }

    public static String encode(String str) throws Exception {
        byte[] encodeBytes = Base64.getEncoder().encode(str.getBytes("utf-8"));
        return new String(encodeBytes);
    }

    public static String decode(String str) throws Exception {
        byte[] decodeBytes = Base64.getDecoder().decode(str.getBytes("utf-8"));
        return new String(decodeBytes);
    }

    public static void main(String[] args){
        System.out.println(new Date(System.currentTimeMillis() + 300*1000));
    }
    /**
     * 生成口令
     * @param loginname 登录名
     * @return String 口令
     */
    public static String sign(String loginname) throws Exception {
        try {
            // 使用私钥进行加密
            String secret = loginname + encode(encryptJWTKey);
            // 设置过期时间:根据当前时间计算出过期时间。此处过期时间是以毫秒为单位,所以乘以1000。
            Date date = new Date(System.currentTimeMillis() + Long.parseLong(accessTokenExpireTime) * 1000);
            // 对私钥进行再次加密
            Algorithm algorithm = Algorithm.HMAC256(secret);
            // 生成token 附带loginname信息
            String token = JWT.create().
                    withClaim("loginname", loginname)
                    .withClaim("currentTimeMillis", System.currentTimeMillis())
                    .withExpiresAt(date)
                    .sign(algorithm);
            return token;
        } catch (Exception e) {
            e.printStackTrace();
            logger.error("JWTToken加密出现异常:" + e.getMessage());
            throw new Exception("JWTToken加密出现异常:" + e.getMessage());
        }

    }

    /**
     * 检验token是否有效
     * @param token 口令
     * @return boolean有效为true
     */
    public static boolean verify(String token) {
        try {
            // 通过token获取登录名
            String secret = getClaim(token, "loginname") + encode(encryptJWTKey);
            // 进行二次加密
            Algorithm algorithm = Algorithm.HMAC256(secret);
            // 使用JWTVerifier进行验证解密
            JWTVerifier verifier = JWT.require(algorithm).build();
            verifier.verify(token);
            return true;
        } catch (Exception e) {
            //e.printStackTrace();
            //4logger.error("JWTToken认证解密出现UnsupportedEncodingException异常:" + e.getMessage());
            return false;
        }
    }

    /**
     * 获取token中的信息就是withClaim中设置的值
     * @param token 口令
     * @param claim sign()方法中withClaim设置的key
     * @return String 值
     */
    public static String getClaim(String token, String claim) throws Exception {
        try {
            // 对token进行解码获得解码后的jwt
            DecodedJWT jwt = JWT.decode(token);
            // 获取到指定的claim,如果是其他类型返回null
            return jwt.getClaim(claim).asString();
        } catch (Exception e) {
            //e.printStackTrace();
            //logger.error("解密Token中的公共信息出现异常:" + e.getMessage());
            throw new Exception("解密Token中的公共信息出现异常:" + e.getMessage());
        }
    }

}

这里面@Value中的${...}是 配置文件application中配置的内容

Token工具类中的常用方法

sign()方法:对指定字段进行加密

  1. 签名生成
    • sign() 方法用于对Token进行签名。它确保Token的内容在传输过程中未被篡改。
    • 通过指定的签名算法(如HMAC SHA256)和密钥对Token的头部和负载部分进行加密,生成一个签名部分。

getClaim()方法

  1. 声明获取
    • getClaim() 方法用于从已解码的JWT中获取特定的声明(Claim)。
    • 它返回一个Claim对象,该对象包含声明的名称和值,可以通过asXXX()方法将其转换为相应的数据类型(如String、Integer等)。

verify()方法

  1. Token验证
    • verify() 方法用于验证Token的合法性。它检查Token的签名是否正确,以确保Token未被篡改且是由可信的签发者签发的。
    • 如果Token合法,该方法将返回一个解析后的JWT对象;如果不合法,则会抛出异常。

Html中的常用方法

1.	SessionStorage: html5的功能,可以在会话期间保存键值对,取得值,删除键值对。
sessionStorage.setItem(key, value) ;  保存
sessionStorage.getItem(key):        取得
sessionStorage.removeItem(key);       删除。
2.	localStorage: html5的功能,在前端浏览器中保存数据,数据会一直存储,直到主动删除。

配置完pom.xml,创建完工具类,便可以使用了,在要使用的地方使用@Autowired注入工具类

使用工具类中的 sign()方法给字段加密,返回前台

此时已经与Redis联动,使用了StringRedisTemPlate,第二行先忽略

返回的ResponseResult涉及到了统一异常处理和响应类

前台:登录成功的话,前端接收后台return的token,并保存键值对。

通过token,以后便可以配置鉴权拦截器了。(都已与Redis联合)

若登录成功,便可以根据token查到对应的数据,若数据为空,说明没登录

便可以使用response.getWriter().write( s: "{\"code\":401}");  给前端页面发送鉴权失败的响应(json格式),前端页面收到401会返回到对应页面

拦截器:

 

Redis

springBoot中集成Jedis,导入依赖

<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>fastjson</artifactId>
    <version>1.2.75</version>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

配置文件中进行相关配置

spring.redis.host=127.0.0.1 (系统有默认值,可以不写)
#Redis服务器连接端口
spring.redis.port=6379 (系统有默认值,可以不写)
#Redis服务器连接密码(默认为空)
spring.redis.password=123456

之后便可以注入,与token一同使用 

标签:return,String,redis,JWT,token,Token,static,使用
From: https://blog.csdn.net/lxy20030530/article/details/143269173

相关文章

  • 关于Rust使用cross进行交叉编译,openssl报错
    遇到错误详情解决:使用cross交叉编译,确保已经安装好了openssl的相关依赖,但是依然报和openssl有关的错误,此时需要在 Cargo.toml 添加openssl的依赖查看:https://github.com/sfackler/rust-openssl?tab=readme-ov-file1[dependencies]2openssl={version="0.10",feat......
  • tmux使用
    使用tmux安装tmux$sudoaptinstall-ytmux启动tmux会话$tmuxnew-s<取个名字>tmux内操作balabalabla退出tmux按Ctrl+b再按d。重新进入tmux列出tmux活动的会话tmuxls重新连接到之前的名为<取个名字>会话tmuxattach-session-t<取个名字>---------------......
  • Python中使用共享变量+信号量实现进程间的实时通信
    【Python程序1中】importctypesimportposix_ipcimportmultiprocessingfrommultiprocessingimportshared_memory#如果系统中已经存在名为/semaphore1的信号量对象,Python并不会重新初始化它,而是使用现有的信号量,#导致现有的信号量可能有残留状态,使得acquire()一直阻......
  • redis主从复制与哨兵机制
    一、主从复制1、主从复制主从复制架构用来备份主节点的数据,主节点接受用户请求,从节点同步数据2、架构图3、搭建主从复制创建三个目录代表三台机器,master-7000,slave-7001,slave-7002拷贝源码中的redis.conf分别到master-7000,slave-7001,slave-7002中修改三台机器......
  • OpenGMS是什么?如何使用OpenGMS的建模与模拟工具(一)
    目录OpenGMS是什么?如何使用OpenGMS的建模与模拟工具(一)一、什么是OpenGMS1、OpenGMS网站 2、OpenGMS团队二、为什么我们需要OpenGMS1、地理模拟实验的局限性区域性限制了科研应用的效率2、外界对于OpenGMS的评价三、 OpenGMS的模型调用方法1、注册账号2、获取需要......
  • 【IntelliJ IDEA】2024最新使用
    大家好!今天我非常高兴能够在这里与大家分享一份极具价值的资源——《IntelliJIDEA2024最新使用》。而IntelliJIDEA,作为业界领先的集成开发环境,以其强大的功能和出色的用户体验,成为了众多开发者的首选。这不仅包括其在代码编辑、调试、版本控制等方面的强大功能,还将涵盖如何......
  • ElasticSearch - Bucket Script 使用指南
    文章目录官方文档BucketScript官文1.什么是ElasticSearch中的BucketScript?2.适用场景3.BucketScript的基本结构4.关键参数详解5.示例官方示例:计算每月T恤销售额占总销售额的比率百分比示例计算:点击率(CTR)6.注意事项与限制7.最佳实践官方文档ht......
  • 【JumpServer教程】简便添加Windows资产:JumpServer堡垒机使用指南
    简介:本文是JumpServer堡垒机使用指南,介绍了如何在JumpServer中简便添加Windows资产的步骤,包括准备工作、开启Windows远程设置、在JumpServer中配置Windows资产以及授权使用。一、背景在很多时候,还有些传统公司,使用的是windowsserver服务器,所以对于这类资产如何管理呢?别急,ju......
  • 深度学习入门笔记——DataLoader的使用
    如何使用数据集DataSet?在介绍DataLoader之前,需要先了解数据集DataSet的使用。Pytorch中集成了很多已经处理好的数据集,在pytorch的torchvision、torchtext等模块有一些典型的数据集,可以通过配置来下载使用。以CIFAR10数据集为例,文档已经描述的很清晰了,其中要注意的就是transform......
  • Qt开发:如何使用QPainter进行2D图形的绘制
    文章目录一、QPainter简介二、如何使用QPainter绘图三、QPainter的绘图区域四、QPainter的常用方法五、QPen的主要功能和属性六、QBrush的主要功能和属性七、QFont的主要功能和属性一、QPainter简介QPainter是Qt框架中的一个强大的绘图类,用于在各种设备上进行2D......