当前位置: 首页 > 编程日记 > 正文

SpringBoot 2.x 使用 JWT(JSON Web Token)

一、跨域认证遇到的问题

由于多终端的出现,很多的站点通过 web api restful 的形式对外提供服务,采用了前后端分离模式进行开发,因而在身份验证的方式上可能与传统的基于 cookieSession Id 的做法有所不同,除了面临跨域提交 cookie 的问题外,更重要的是,有些终端可能根本不支持 cookie

JWT(JSON Web Token) 是一种身份验证及授权方案,简单的说就是调用端调用 api 时,附带上一个由 api 端颁发的 token,以此来验证调用者的授权信息。

一般流程是下面这样:

1. 用户向服务器发送用户名和密码。
2. 服务器验证通过后,在当前对话(session)里面保存相关数据,比如用户角色、登录时间等等。
3. 服务器向用户返回一个 session_id,写入用户的 Cookie。
4. 用户随后的每一次请求,都会通过 Cookie,将 session_id 传回服务器。
5. 服务器收到 session_id,找到前期保存的数据,由此得知用户的身份。

这种模式的问题在于扩展性不好。单机没有问题,如果是服务器集群、跨域的服务导向架构或者用户禁用了 cookie ,就不行了。

二、解决方案

1. 单机和分布式应用下登录校验,session 共享

  • 单机和多节点 tomcat 应用登录检验

    ①、单机 tomcat 应用登录,sesssion 保存在浏览器和应用服务器会话之间,用户登录成功后,服务端会保证一个 session,也会给客户端一个 sessionId,客户端会把 sessionId 保存在 cookie 中,用户每次请求都会携带这个 sessionId

    ②、多节点 tomcat 应用登录,开启 session 数据共享后,每台服务器都能够读取 session。缺点是每个 session 都是占用内存和资源的,每个服务器节点都需要同步用户的数据,即一个数据需要存储多份到每个服务器,当用户量到达百万、千万级别的时,占用资源就严重,用户体验特别不好!!

  • 分布式应用中 session 共享

    ①、真实的应用不可能单节点部署,所以就有个多节点登录 session 共享的问题需要解决。tomcat 支持 session 共享,但是有广播风暴;用户量大的时候,占用资源就严重,不推荐

    ②、Reids 集群,存储登陆的 token,向外提供服务接口,Redis 可设置过期时间(服务端使用 UUID生成随机 64 位或者 128token ,放入 Redis 中,然后返回给客户端并存储)。

    ③、用户第一次登录成功时,需要先自行生成 token,然后将 token 返回到浏览器并存储在 cookie 中,
    并在 Redis 服务器上以 tokenkey,用户信息作为 value 保存。后续用户再操作,可以通过 HttpServletRequest 对象直接读取 cookie 中的 token,并在 Redis 中取得相对应的用户数据进行比较(用户每次访问都携带此 token,服务端去 Redis 中校验是否有此用户即可)。

    ④、 缺点:必须部署 Redis,每次必须访问 RedisIO 开销特别大。

2. 最终解决方案:使用 JWT 实现 Token 认证

  • JWT 的原理

    服务器认证以后,生成一个 JSON 对象发回给用户,以后用户与服务端通信的时候,都要发回这个 JSON 对象。服务器完全只靠这个对象认定用户身份。为了防止用户篡改数据,服务器在生成这个对象的时候,会加上签名。也就是说服务器就不保存任何 session 数据了,即服务器变成无状态了,从而比较容易实现扩展。

    简单来说,就是通过一定规范来生成 token,然后可以通过解密算法逆向解密 token,这样就可以获取用户信息
  • 优点和缺点

    优点:生产的 token 可以包含基本信息,比如 id、用户昵称、头像等信息,避免再次查库;存储在客户端,不占用服务端的内存资源

    缺点:token 是经过 base64 编码,所以可以解码,因此 token 加密前的对象不应该包含敏感信息(如用户权限,密码等)

  • JWT 格式组成:头部+负载+签名 ( header + payload + signature )

    头部:主要是描述签名算法。

    负载:主要描述是加密对象的信息,如用户的 id 等,也可以加些规范里面的东西,如 iss 签发者,exp 过期时间,sub 面向的用户。

    签名:主要是把前面两部分进行加密,防止别人拿到 token 进行base 解密后篡改 token。

3. 案例图设计

1269192-20190901141638895-414417269.png

三、代码演示案例

  • pom.xml 文件引入依赖和实体类
    <!-- 依赖可以减少实体类 getter/setter等方法书写 -->
    <dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><optional>true</optional>
    </dependency>
    <!-- JWT相关 -->
    <dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt</artifactId><version>0.7.0</version>
    </dependency>====================================================================================@Getter
    @Setter
    @ToString
    @AllArgsConstructor
    @NoArgsConstructor
    public class User implements Serializable {private Integer id;private String openid;private String name;private String headImg;private String phone;private String sign;private Integer sex;private String city;private Date createTime;}
  • 生成 JWT 工具类
    public class JwtUtil {// 主题public static final String SUBJECT = "RookieLi";// 秘钥public static final String SECRETKEY = "Rookie666";// 过期时间public static final long EXPIRE = 1000 * 60 * 60 * 24 * 7;  //过期时间,毫秒,一周// 生成 JWTpublic static String geneJsonWebToken(User user) {if (user == null ||user.getId() == null ||user.getName() == null ||user.getHeadImg() == null) {return null;}String token = Jwts.builder().setSubject(SUBJECT).claim("id", user.getId()).claim("name", user.getName()).claim("img", user.getHeadImg()).setIssuedAt(new Date()).setExpiration(new Date(System.currentTimeMillis() + EXPIRE)).signWith(SignatureAlgorithm.HS256, SECRETKEY).compact();return token;}// 校验 JWTpublic static Claims checkJWT(String token) {try {final Claims claims = Jwts.parser().setSigningKey(SECRETKEY).parseClaimsJws(token).getBody();return claims;} catch (Exception e) {e.printStackTrace();}return null;}
    }
    
  • 测试 JWT 工具类
    public class JwtUtilTest {@Testpublic void testGeneJwt(){User user = new User();user.setId(999);user.setHeadImg("I'm busy");user.setName("Rookie");String token = JwtUtil.geneJsonWebToken(user);System.out.println(token);}@Testpublic void testCheck(){// 下面此 token 字符串是上面的结果生成的,每次不一样,不是写死的String token = "eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJSb29raWVMaSIsImlkIjo5OTksIm5hbWUiOiJSb29raWUiLCJpbWciOiJJJ20gYnVzeSIsImlhdCI6MTU2NzMxNjk4NywiZXhwIjoxNTY3OTIxNzg3fQ.FJh41VwVh2gh5-_cOG0SOgoO3dR_ZcK9VWNNskWqKl0";Claims claims = JwtUtil.checkJWT(token);if(claims != null){String name = (String)claims.get("name");String img = (String)claims.get("img");int id =(Integer) claims.get("id");System.out.println(name);System.out.println(img);System.out.println(id);}else{System.out.println("非法token");}}
    }

    参考博客:

    http://www.ruanyifeng.com/blog/2018/07/json_web_token-tutorial.html

    https://www.cnblogs.com/jpfss/p/10929458.html

转载于:https://www.cnblogs.com/miantiao312/p/11442187.html

相关文章:

在Unity中制作4种不同的游戏

流派:电子学习| MP4 |视频:h264&#xff0c;1280720 |音频:AAC&#xff0c;48.0 KHz 语言&#xff1a;英语中英文字幕&#xff08;根据原英文字幕机译更准确&#xff09;|大小解压后:8.6 GB 含课程素材 |时长:15h 3m Unity 制作4款无代码手机游戏 Make 4 games in Unity with …

Spring学习笔记:1(初步认识概念)

Spring的三大主要特征 spring主要特征有三个&#xff1a;控制反转&#xff08;IOC&#xff09;&#xff0c;依赖注入&#xff08;DI&#xff09;和面向切面&#xff08;AOP&#xff09;。 IoC&#xff1a;Inverse of Control&#xff08;控制反转&#xff09; 1.对控制反转的…

shell example01

条件判断 if [[ -e ${1} ]]; thenecho "$(tput setaf 2) found ${1} $(tput sgr0)"cat ${1} elseecho "$(tput setaf 1) not found ${1} $(tput sgr0)"exit 1 fi//简化[[ -e ${1} && -e ${2} ]] && cat ${1} > ${2}//判断取反txt4.txti…

gradle教程 [原创](eclipse/ADT下 非插件 非Android Studio/AS)纯手打 第一篇:安装配置gradle...

一个bug 一个脚印的叫你们用gradle。 1介于网络上的很多资料都是老的 不适用与现在的新版本gradle 尤其是有些gradle方法改名了老的用不了 2介于网上都是粘贴复制并且零碎我很蛋疼啊&#xff0c;走了很多歪路才弄出来&#xff0c;所以我弄一个完全完整的版本 3我不但会写gradle…

java的static关键字

java的static关键字 静态变量和静态方法 static关键字最基本的用法是&#xff1a; 1、被static修饰的变量属于类变量&#xff0c;可以通过类名.变量名直接引用&#xff0c;而不需要new出一个类来 2、被static修饰的方法属于类方法&#xff0c;可以通过类名.方法名直接引用&…

Spring学习笔记:2(IOC装配Bean之xml方式)

xml配置方式装配Bean 本文借鉴于&#xff1a;https://www.cnblogs.com/qdhxhz/p/6511887.html Spring框架Bean实例化的方式提供了三种方式实例化Bean 构造方法实例化(默认无参数&#xff0c;用的最多)静态工厂实例化实例工厂实例化 代码如下&#xff1a; Bean1类(构造方法…

学习RPG Maker MZ开发创建并发布PC和移动端游戏

Complete RPG Maker MZ: Create and Publish for PC and Mobile 完整的RPG制造商MZ:为个人电脑和移动设备创建和发布 MP4 |视频:h264&#xff0c;1280720 |音频:AAC&#xff0c;44.1 KHz&#xff0c;2 Ch 语言&#xff1a;英语中英文字幕&#xff08;根据原英文字幕机译更准确…

CSS选择器总结

总结几种自己比较容易混淆的&#xff1a; 1. 后代选择器&#xff0c;写法是 E1 E2&#xff0c;如 ul li&#xff0c;选择的是所有后代&#xff0c;包括子后代、孙后代…; 2. 子选择器&#xff0c;写法 E1 > E2&#xff0c;只选择子后代&#xff0c;不包括孙后代元素&#xf…

OSChina 周六乱弹 —— 小明和网关超经典的故事~

2019独角兽企业重金招聘Python工程师标准>>> 周六&#xff0c;又到了瞎扯淡的时间了。周末&#xff0c;约会&#xff0c;男男女女&#xff0c;还有那啥那啥&#xff0c;你们懂得&#xff5e; 男人和女人明显不同&#xff0c;这样才导致了异性相吸吗&#xff1f; 1. …

概念艺术绘画学习教程 Schoolism – Foolproof Concept Painting with Airi Pan

Schoolism——万无一失的概念绘画潘 大小解压后&#xff1a;3.19G 含课程素材文件 1920X1080 .mp4 语言&#xff1a;英语中英文字幕&#xff08;根据原英文字幕机译更准确&#xff09; 信息: 万无一失的概念绘画潘 本课程由概念设计师兼插画师潘开发&#xff0c;与大家分享她…

Mybatis复习笔记:1

关于模糊查找 模糊查找其实有两种基本操作(之前学的时候看的不太仔细&#xff0c;漏了…) 第一种 <select id"findByType" parameterType"String" resultType"com.domain.User">select * from product where product_type like #{produ…

文件只能安装一次

1 def get_mac_address(): 2 macuuid.UUID(int uuid.getnode()).hex[-12:] 3 return ":".join([mac[e:e2] for e in range(0,11,2)]) 获取计算机名字 1 import socket 2 socket.gethostname() linux下获取主机外网ip 1 import socket2 import fcntl3 impo…

Linux pipe函数

1. 函数说明 pipe&#xff08;建立管道&#xff09;&#xff1a; 1) 头文件 #include<unistd.h> 2) 定义函数&#xff1a; int pipe(int filedes[2]); 3) 函数说明&#xff1a; pipe()会建立管道&#xff0c;并将文件描写叙述词由參数filedes数组返回。 fi…

操作系统知识点:全面

操作系统知识点&#xff1a;全面 https://www.jianshu.com/p/c3a3cc0254b1 https://www.jianshu.com/u/881ef7b85f62 posted on 2019-09-03 21:44 竹径风声 阅读(...) 评论(...) 编辑 收藏 转载于:https://www.cnblogs.com/girl1314/p/11455906.html

在UE4中创建CG动画 How to create a movie in Unreal Engine 4 using Metahuman

MP4 |视频:h264&#xff0c;1280720 |音频:AAC&#xff0c;44.1 KHz 语言&#xff1a;英语中英文字幕&#xff08;根据原英文字幕机译更准确&#xff09;|大小解压后:1.55 GB |时长:1h 16m 你会学到什么 如何在虚幻引擎4中创建CG动画 虚幻引擎4 Metahuman 使用metahuman在虚幻引…

MyBatis复习笔记2:配置文件详解

配置文件详解 属性&#xff08;properties&#xff09; MyBatis可以使用 properties 来引入外部 properties 配置文件的内容 resource&#xff1a;引入类路径下的资源 url&#xff1a;引入网络路径或者磁盘路径下的资源 properties 有三种方式使用&#xff1a; 1、通过 reso…

Android ActionBarDrawerToggle、DrawerLayout、ActionBar 结合

ActionBarDrawerToggle是一个开关。用于打开/关闭DrawerLayout抽屉 ActionBarDrawerToggle 提供了一个方便的方式来配合DrawerLayout和ActionBar。以实现推荐的抽屉功能。 即点击ActionBar的homebutton&#xff0c;就可以弹出DrawerLayout抽屉。 在Activity中的两个回调函数中使…

【转】statfs获得硬盘使用情况 模拟linux命令 df

原文网址&#xff1a;http://blog.csdn.net/mociml/article/details/5335474 说明&#xff1a;本文以主要为转载内容&#xff0c;同时加入了我在使用过程中遇到问题对其的修正&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff0…

SQL常见的面试题

SQL常见的面试题 https://www.jianshu.com/p/558f2113bb62 posted on 2019-09-03 21:55 竹径风声 阅读(...) 评论(...) 编辑 收藏 转载于:https://www.cnblogs.com/girl1314/p/11455959.html

Mybatis入门:4(多表查询操作)

多表查询操作 Mybatis的多表操作 表之间的关系有几种&#xff1a;一对多、一对一、多对一、多对多 举例: 用户和订单就是一对多——一个用户可以下多个订单 订单和用户就是多对一——多个订单属于同一个用户 人和身份证号就是一对一 一个人只能有一个身份证号 一个身份证号只…

Python for虚幻引擎编辑器工具脚本学习教程

Python for Unreal Engine Editor Tools Scripting MP4 |视频:h264&#xff0c;1280720 |音频:AAC&#xff0c;44.1 KHz&#xff0c;2 Ch 语言&#xff1a;英语中英文字幕&#xff08;根据原英文字幕机译更准确&#xff09; |时长:23节课(4h 8m) |大小解压后:2.7 GB 含课程文件…

C++ Windows进程管理

功能&#xff1a; 1.各个进程启动、挂起、恢复、停止等 2.监听进程的运行状态&#xff0c;进程退出&#xff08;正常、非正常&#xff09;时&#xff0c;通知用户 3.异步队列 4.线程安全 进程管理器类&#xff1a; #ifndef __ProcessManager_ProcessManager_H__ #define __Proc…

shell中和||的使用方法

&&运算符:command1 && command2&&左边的命令&#xff08;命令1&#xff09;返回真(即返回0&#xff0c;成功被执行&#xff09;后&#xff0c;&&右边的命令&#xff08;命令2&#xff09;才能够被执行&#xff1b;换句话说&#xff0c;“如果…

SQL中内连接、外连接、交叉连接

SQL中内连接、外连接、交叉连接 SQL连接可以分为内连接、外连接、交叉连接。 数据库数据&#xff1a; book表 stu表 1.内连接 1.1.等值连接&#xff1a;在连接条件中使用等于号()运算符比较被连接列的列值&#xff0c;其查询结果中列…

Blender从头开始装配和动画制作低多边形风格的FPS手臂

Rigging and Animating Low Poly FPS Arms in Blender MP4 |视频:h264&#xff0c;1280720 |音频:AAC&#xff0c;44.1 KHz&#xff0c;2 Ch 语言&#xff1a;英语中英文字幕&#xff08;根据原英文字幕机译更准确&#xff09; |时长:21节课(4h 56m) |大小解压后:3.16 GB 含课程…

Mybatis复习笔记3:映射文件详解

映射文件详解 参数处理&#xff08;#和$的区别&#xff09; #{}&#xff1a;可以获取map中的值或者实体对象属性的值&#xff1b;${}&#xff1a;可以获取map中的值或者实体对象属性的值&#xff1b; select * from person where id${id} and name#{name} # 控制台输出&…

TMS320F28335项目开发记录2_CCS与JTAG仿真器连接问题汇总

CCS与仿真器连接问题 实际使用过程中。仿真器和CCS连接可能出现这样或那样的问题&#xff0c;或许你的连接非常成功&#xff0c;没碰到过什么问题。但我的问题的确不少&#xff0c;可能与电脑配置有关吧&#xff0c;也可能与人品有关吧。 以下的自己的一些错误和解决方法总…

Mysql备份与还原及优化方法

Mysql备份一般采用mysqldump命令&#xff0c;命令形式一般如下&#xff1a;$ mysqldump –hhostname –uuser –ppassword–Pport db_name > db_name.sql默认情况下&#xff0c;不备份存储过程和函数&#xff0c;若要备份存储过程和函数&#xff0c;要加上-R选项&#xff0c…

HashTable和HashMap的区别详解

HashTable和HashMap的区别详解 一、HashMap简介 HashMap是基于哈希表实现的&#xff0c;每一个元素是一个key-value对&#xff0c;其内部通过单链表解决冲突问题&#xff0c;容量不足&#xff08;超过了阀值&#xff09;时&#xff0c;同样会自动增长。 HashMap是非线程安全的&…

Unity电子游戏优化终极指南 The Ultimate Guide to Video Game Optimisation

大小解压后&#xff1a;5.2G 含课程文件 时长9h 1280X720 MP4 语言&#xff1a;英语中英文字幕&#xff08;根据原英文字幕机译更准确&#xff09; 电子游戏优化终极指南 信息: 学会从你的Unity游戏开发项目中挤出每一帧表现 你会学到什么 –如何为游戏制定绩效预算并坚持下…