jwt获取token_JWT实现token认证

1.JWT是什么?

Json Web Token(JWT)是目前比较流行的跨域认证解决方案,是一种基于JSON的开发标准,由于数据是可以经过签名加密的,比较安全可靠,一般用于前端和服务器之间传递信息,也可以用在移动端和后台传递认证信息。

2.JWT的组成

让我们先来看看jwt的实际例子:

Bearer eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJyb2wiOiJST0xFXzEiLCJpc3MiOiJTbmFpbENsaW1iIiwiaWF0IjoxNTk1OTkzMDQwLCJzdWIiOiIxNTAwMDAwMDAwMCIsImV4cCI6MTU5NjU5Nzg0MH0.Wxet2qPG1ajyG9CcQCN29gfKJzal0vCmQSXrwAgQwcI

我们可以看到,这个JWT包含3部分,是由“.”号分开的,Bearer是jwt中的一种前缀规范。

第一部分是Header:

eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9

第二部分是Payload:

eyJyb2wiOiJST0xFXzEiLCJpc3MiOiJTbmFpbENsaW1iIiwiaWF0IjoxNTk2MDAzODM4LCJzdWIiOiIxNTAwMDAwMDAwMCIsImV4cCI6MTU5NjYwODYzOH0

第三方部分是签名Signature:

MJmTSGBM9wRVOY40ORZuSGpT7FQNUcD2JlJxL0K0-YM

3.JWT的流程

先上图:

9edad30454d42e983004a1322e9fe4ae.png

  1. 客户端登录接口输入用户名和密码提交到服务器;

  2. 服务器端校验用户名和密码,校验通过则创建一个JWT;

  3. 服务器端返回该JWT给客户端;

  4. 客户端每请求一次接口在Headers中带上返回的JWT;

  5. JWT自身的算法校验该JWT;

  6. 如果校验通过,操作DB,返回数据;反之返回状态码和错误信息。

4.SpringBoot集成JWT,代码实现

引入jwt的依赖库:

        <dependency>            <groupId>io.jsonwebtokengroupId>            <artifactId>jjwt-apiartifactId>            <version>0.10.7version>        dependency>        <dependency>            <groupId>io.jsonwebtokengroupId>            <artifactId>jjwt-implartifactId>            <version>0.10.7version>            <scope>runtimescope>        dependency>        <dependency>            <groupId>io.jsonwebtokengroupId>            <artifactId>jjwt-jacksonartifactId>            <version>0.10.7version>            <scope>runtimescope>        dependency>

JwtTokenUtils类:用于生成token

/** * @author kevincow */public class JwtTokenUtils {    /**     * 生成足够的安全随机密钥,以适合符合规范的签名     */    private static byte[] apiKeySecretBytes = Base64.getDecoder().decode(SecurityConstants.JWT_SECRET_KEY);    private static SecretKey secretKey = Keys.hmacShaKeyFor(apiKeySecretBytes);    public static String createToken(String username, List<String> roles) {        long expiration = SecurityConstants.EXPIRATION_REMEMBER;        String tokenPrefix = Jwts.builder()                .setHeaderParam("typ", SecurityConstants.TOKEN_TYPE)                .signWith(secretKey, SignatureAlgorithm.HS256)                .claim(SecurityConstants.ROLE_CLAIMS, String.join(",", roles))                .setIssuer("SnailClimb")                .setIssuedAt(new Date())                .setSubject(username)                .setExpiration(new Date(System.currentTimeMillis() + expiration * 1000))                .compact();        return SecurityConstants.TOKEN_PREFIX + tokenPrefix;    }}

接下来需要写一个过滤器去实现token的校验功能:

/** * 实现token的校验功能 * * 过滤器处理所有HTTP请求,并检查是否存在带有正确令牌的Authorization标头。例如,如果令牌未过期或签名密钥正确。 * * @author kevincow */public class JWTAuthorizationFilter extends BasicAuthenticationFilter {    private static final Logger logger = Logger.getLogger(JWTAuthorizationFilter.class.getName());    public JWTAuthorizationFilter(AuthenticationManager authenticationManager) {        super(authenticationManager);    }    @Override    protected void doFilterInternal(HttpServletRequest request,                                    HttpServletResponse response,                                    FilterChain chain) throws IOException, ServletException {        logger.info("token"+request.getHeader(SecurityConstants.TOKEN_HEADER));        String authorization = request.getHeader(SecurityConstants.TOKEN_HEADER);        // 如果请求头中没有token信息则直接放行了        if (authorization == null || !authorization.startsWith(SecurityConstants.TOKEN_PREFIX)) {            chain.doFilter(request, response);            return;        }        // 如果请求头中有token,则进行解析,并且设置授权信息        SecurityContextHolder.getContext().setAuthentication(getAuthentication(authorization));        super.doFilterInternal(request, response, chain);    }    /**     * 获取用户认证信息 Authentication     */    private UsernamePasswordAuthenticationToken getAuthentication(String authorization) {        String token = authorization.replace(SecurityConstants.TOKEN_PREFIX, "");        try {            String username = JwtTokenUtils.getUsernameByToken(token);            logger.info("checking username:" + username);            // 通过 token 获取用户具有的角色            List userRolesByToken = JwtTokenUtils.getUserRolesByToken(token);            if (!StringUtils.isEmpty(username)) {                return new UsernamePasswordAuthenticationToken(username, null, userRolesByToken);            }        } catch (SignatureException | ExpiredJwtException | MalformedJwtException | IllegalArgumentException exception) {            logger.warning("Request to parse JWT with invalid signature . Detail : " + exception.getMessage());        }        return null;    }}

最后写业务实现代码了(这里只展示控制层代码,业务代码就不一一赘述)

/**     * 微信端登录接口,在header中带上token     *     * @param userName     * @param passWord     * @return 是否登录成功     */    @ApiOperation(value = "用户登录成功之后,在header中带上token",            notes = "校验用户名密码,校验完成之后,在header中带上token", httpMethod = "GET")    @PostMapping("/whetherIsSysUser")    public SuccessResponse whetherRegister(@ApiParam("用户名") @RequestParam String userName,                                           @ApiParam("密码") @RequestParam String passWord,                                           HttpServletResponse response) {        try {            User user = userService.whetherRegister(userName, passWord);            if (StringUtils.isEmpty(user)) {                return SuccessResponse.of(false, "该用户未注册");            }            JwtUser jwtUser = new JwtUser(user);            List roles = jwtUser.getAuthorities()                    .stream()                    .map(GrantedAuthority::getAuthority)                    .collect(Collectors.toList());            String token = JwtTokenUtils.createToken(user.getPhone(), roles);            // Http Response Header 中返回 Token            response.setHeader(SecurityConstants.TOKEN_HEADER, token);        } catch (RuntimeException e) {            e.printStackTrace();            return SuccessResponse.of(false, "该用户未注册");        }        return SuccessResponse.of(true);    }

至此,代码实现部分差不多就完成了。接下来我们看下效果:

a80c4f0bcbcbe49b492951bfd0befa09.png

上图可以看到,登录接口用户名密码校验通过之后在headers中返回了Authorization。

然后当客户端请求其他接口并在请求头带上登录时返回的Authorization,就能正常响应了。

如果没有带上Authorization呢?那么会报一个错。

{    "timestamp": "2020-07-29 15:38:23",    "status": 401,    "error": "Unauthorized",    "message": "Full authentication is required to access this resource",    "path": "/wx/users/showNewVersion"}

以上就差不多是jwt的整个流程了。

5.总结

    从特点中看优点:

  • JSON的通用性,决定了JWT是个跨语言的技术;

  • JWT token对于sessionid的方案来说,JWT可以携带任何我们想要用到的信息;

  • 安全性高,防止token伪造和篡改;

  • JWT token是自校验的形式,不需要任何其他请求和数据库操作,使我们系统管理会话更高效;

  • JWT不需要在服务端保存会话信息, 所以它易于应用的扩展

    从特点中找缺点:

  • 一旦成功签发JWT token,无法手动将其过期;

  • 在token签发后的有效时间内,JWT无法做到及时获取最新数据,例如修改密码后无感知;

  • 存储空间、网络流量开销更大;

  • 为了安全,JWT需要使用https协议,以免token泄露

以上。

我是凯文cow,念念不忘,必有回响。

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

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

相关文章

神经网络与深度学习——TensorFlow2.0实战(笔记)(三)(第一个Python程序)

从程序中了解知识点&#xff0c;有详细的备注 # 判断变量num是否为正数 # 注释语句 是对程序进行说明的语句&#xff0c;在程序运行过程中不被执行 # ctrl/进行多行注释num0 # 赋值语句 大小写敏感 可以以分号结尾&#xff0c;但是在同一行必须以分号隔开 # 其数据类型由所赋…

Sqlce与SQL Server2000/2005数据转换程序

先上图 功能介绍&#xff1a; 1.根据Mssql2000/2005创建sqlce数据库表结构及索引(如果操作的表为空&#xff0c;则取mssql2000/2005的所有表&#xff0c;否则取录入的操作的表&#xff0c;用逗号隔开) 2.从mssql2000/2005导数据到已建好的sqlce(如果操作的表为空&#xff0c;则…

docker java镜像_Springboot整合MongoDB的Docker开发,其它应用也类似

1 前言Docker是容器开发的事实标准&#xff0c;而Springboot是Java微服务常用框架&#xff0c;二者必然是会走到一起的。本文将讲解如何开发Springboot项目&#xff0c;把它做成Docker镜像&#xff0c;并运行起来。2 把Springboot打包成Docker镜像Springboot的Web开发非常简单&…

神经网络与深度学习——TensorFlow2.0实战(笔记)(三)(python输入和输出)

从程序中学习知识点 1. #输出函数可以输出字符串常量 print("hello") #输出数学表达式 print(12) #输出变量 print(num) 2. #string input(提示信息) #接收用户的输入&#xff0c;并以字符串类型返回;即键盘输入123&#xff0c;并将123赋值给string numint(input…

ManulResetEvent与AutoResetEvent

定义 ManualResetEvent&#xff1a;通知正在等待的线程已发生事件。 AutoResetEvent&#xff1a;通知正在等待的线程已发生事件。 代码 ManualResetEvent测试 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading;…

神经网络与深度学习——TensorFlow2.0实战(笔记)(三)(python常量、变量和表达式)

从程序中学习知识点 1. #支持6中表达形式 数字 字符串 列表 元组 字典 集合 #数字 整型(正整数 负整数 零) #Python3中的整数可以任意大&#xff0c;而不用担心位数不够而导致溢出的情况 intnum12345678909999999999999 print(intnum,type(intnum)) #浮点数 小数 floatnum1 …

reentrantlock非公平锁不会随机挂起线程?_程序员必须要知道的ReentrantLock 及 AQS 实现原理...

专注于Java领域优质技术&#xff0c;欢迎关注作者&#xff1a;Float_Luuu提到 JAVA 加锁&#xff0c;我们通常会想到 synchronized 关键字或者是 Java Concurrent Util(后面简称JCU)包下面的 Lock&#xff0c;今天就来扒一扒 Lock 是如何实现的&#xff0c;比如我们可以先提出一…

excel办公常用的宏_让领导看傻!精美168套办公常用excel模板免费领

HR们面试的时候&#xff0c;是不是经常看到应聘者的简历上技能那一栏写着精通Excel、PPT等办公技能&#xff1f;你知道Excel用到什么程度才算精通吗&#xff1f;能够用excel做表格就算精通吗&#xff1f;还是要能够熟练使用各种函数&#xff1f;你做出来的Excel报表也许是这样的…

神经网络与深度学习——TensorFlow2.0实战(笔记)(三)(python运算符和表达式)

从程序中学习知识点 1.算术运算符 #运算符&#xff08;Operator&#xff09;&#xff1a;完成不同类型的常量、变量之间的运算 #除法运算 / 结果是一个浮点型的精确数的值&#xff0c;与java等其他语言的不同之处 print(7/2,7.0/2,-7/2) #整除运算 print(7//2,-7//2) print(7…

float32精度_PyTorch 1.6来了:新增自动混合精度训练、Windows版开发维护权移交微软...

刚刚&#xff0c;Facebook 通过 PyTorch 官方博客宣布&#xff1a;PyTorch 1.6 正式发布&#xff01;新版本增加了一个 amp 子模块&#xff0c;支持本地自动混合精度训练。Facebook 还表示&#xff0c;微软已扩大了对 PyTorch 社区的参与&#xff0c;现在拥有 PyTorch 在 Windo…

神经网络与深度学习——TensorFlow2.0实战(笔记)(三)(python语句)

1.if语句 #if语句 x,y 3,5 if x<y:print("x<y") elif xy:print("xy") else:print("x>y") 2.条件表达式 x,y3,5 #表达式1(条件为真的结果) if 判断条件 else 表达式2(条件为假的结果) print(x if x>y else y) 3.while语句 #死循环…

天气预报的获取

好久没有写技术文章了&#xff0c;2010年工作太忙&#xff0c;奔波在国内各地&#xff0c;也没什么时间关注一些技术方面的事情&#xff0c;最近有一个项目封闭开发&#xff0c;有些时间来写些琐碎的东西&#xff0c;就当是整理下最近的东西吧&#xff0c;没什么技术价值&#…

神经网络与深度学习——TensorFlow2.0实战(笔记)(四)(python列表与元组)

序列数据结构 1.成员是有序排列的 2.每个元素的位置称为下标或索引 3.通过索引访问序列中的成员 4.Python中的序列数据类型有字符串、列表、元组 "abc" ≠ "bca" 5.Python中的列表和元组&#xff0c;可以存放不同类型的数据 列表使用方括号[ ]表示&a…

apktoolkit apk反编译没有文件_[工具] Mac下一键APK逆向环境

安装apktool和dex2jar,jd-guihomebrew安装&#xff1a; brew install apktool brew install dex2jar JD-GUI去http://jd.benow.ca/下载 dmg可能不支持最新版本的mac用不了&#xff0c;打开就报错反编译流程执行脚本apktool d xxx.apk 注&#xff1a;xxx.apk为你要反编译的apk…

搭建你的嵌入式Vxworks开发环境

3.1 最常见的开发环境配置使用串口和网络连接&#xff08;host和target之间&#xff09;。串口连接用于和boot loader之间的通信&#xff08;如输出信息在host上的显示&#xff09;&#xff0c;网络连接用于传输文件&#xff0c;包括Vxworks system image。默认情况下使用网络连…

神经网络与深度学习——TensorFlow2.0实战(笔记)(四)(python字典和集合)

字典和集合 字典 每个字典元素都是一个键(关键字)/值(关键字对应的取值)对 #创建字典 dic_score{"语文":80,"数学":99} #打印 print(dic_score) print(dic_score["语文"]) #长度 print(dic_score.__len__)#错误写法 print(len(dic_score)) #…

[Project Euler]加入欧拉 Problem 9

A Pythagorean triplet is a set of three natural numbers, a b c, for which, a2 b2 c2 For example, 32 42 9 16 25 52. There exists exactly one Pythagorean triplet for which a b c 1000.Find the product abc. 意思很明白找出这样的a, b, c abc 1000 a …

java 在已有的so基础上封装jni_[干货]再见,Android JNI 封装

1 前言2 JNI 速查表2.1 Java 和 Native 数据类型映射表2.2 引用类型3 JNI 理论基础速览4 JNI 常用场景示例4.1 字符串传递(java->native)4.2 字符串返回(native->java)4.3 数组传递(java->native)4.4 其他复杂对象传递(java->native)4.5 复杂对象返回(native->j…

神经网络与深度学习——TensorFlow2.0实战(笔记)(四)(python函数)

函数&#xff08;function) &#xff1a;实现某种特定功能的代码块 优点&#xff1a;程序简洁&#xff0c;可重复调用、封装性好、便于共享 类别&#xff1a;系统函数和用户自定义函数 Python内置函数 数学运算函数 print(abs(-1)) print(pow(2,3)) print(round(3.1415926,…

神经网络与深度学习——TensorFlow2.0实战(笔记)(四)(python模块、包和库)

模块&#xff08;Module&#xff09; 模块是一个python文件&#xff08;.py&#xff09;&#xff0c;拥有多个功能相近的函数或类。 便于代码复用&#xff0c;提高编程效率&#xff0c;提高了代码的可维护性。 避免函数名和变量名冲突。 包&#xff08;Package&#xff09;…