移动端前后端分离实现Token认证

本文介绍了使用JWT进行移动端登录认证的实现过程,包括JWT的结构、生成与使用,以及在实际项目中如何配置和处理续签、注销等问题。通过结合redis缓存和拦截器,实现了安全的无状态RESTful API交互。
摘要由CSDN通过智能技术生成

简介

         在实际项目开发当中,如果做微信小程序h5等项目时,需求需要用到token做登陆认证时,大家可能都会使用数据库存储token手动随机生成token凭证,今天教大家使用jwt做token

认证,采用的技术,redis缓存,jwt,拦截器,过滤器,会详细讲解jwt优缺点后续解决办法。

 

jwt介绍

        JWT 全称 JSON Web Tokens ,是一种规范化的 token。是对 token 这一技术提出一套规范。

 

JWT的结构

    头部(header)
    载荷(Payload)
    签名(Signature)

 

Header

头部包含了两部分,采用的签名算法和token 类型

{
  "alg": "HS256",
  "typ": "JWT"
}

 

Payload

这部分就是我们存放信息的地方了,你可以把用户 ID 等信息放在这里(当然用户ID推荐加密后存放),JWT 规范里面对这部分有进行了比较详细的介绍,常用的由 iss(签发者),iat(签发时间),exp(过期时间),sub(面向的用户),aud(接收方)

{
    "iss": "lion1ou JWT",
    "iat": 1441593502,
    "exp": 1441594722,
    "aud": "www.example.com",
    "sub": "example@163.com"
}

 

Signature

Header 和 Payload 编码后的字符串拼接后再用HS256签名算法(Header中alg指明的算法)加密,在加密的过程中还需加上secret(密钥),最后得到签名

Tips:Base64是一种编码,并不是一种加密过程,不具备安全性,所以敏感信息别放入其中

 

JWT 生成 TOKEN

  • header json 的 base64 编码为令牌第一部分
  • payload json 的 base64 编码为令牌第二部分
  • 拼装第一、第二部分编码后的 json 以及 secret 进行签名的令牌的第三部分
// 头部
header = '{"alg":"HS256","typ":"JWT"}'

// 载荷
payload = '{"loggedInAs":"admin","iat":1422779638}'

// 对 header 和 payload 分别进行 base64 编码后拼接起来进行 hash 运算得到签名
signature = HMAC-SHA256(encodeBase64Url(header) + "." + encodeBase64Url(payload),secret)

// 最后将三部分进行 base64 编码后再拼接即可得到 token。
json_web_token = encodeBase64Url(header) + '.' + encodeBase64Url(payload) + '.' + encodeBase64Url(signature)

JWT 使用

请求

  • url 参数:?token=你的token
  • header 中:Authorization:Bearer <token>(推荐)

注意Bearer和token之间有一个空格

接收

前端接收到后端返回的 token 后进行存储,一般存储在:

  1. LocalStorage
  2. SessionStorage
  3. Cookie

 

JWT 的优势和问题

优势

  1. 防 CSRF
  2. 适合移动应用
  3. 无状态的 RESTful API
  4. 一次性验证(如邮件激活)

 

问题

  1. 续签问题
  2. 注销问题

 

JWT 安全

  • 使用 HTTPS
  • 缩短 token 有效时间
  • 定期更新密钥
  • 使用HttpOnly属性来防止Cookie被JavaScript读取,防止 XSS 攻击窃取 cookie
  • 不要在token中编码敏感信息,如果一定会有那必须先加密再进行编码
  • 根据需要可将用户 IP 放入 Payload 进行校验

    关于jwt的token的续签问题,注销问题后续代码体现于解决办法,下面代码体现。

 

一编写JWT的Toke工具包

import com.auth0.jwt.JWT;
import com.auth0.jwt.JWTVerifier;
import com.auth0.jwt.algorithms.Algorithm;
import com.auth0.jwt.exceptions.TokenExpiredException;
import com.auth0.jwt.interfaces.DecodedJWT;
import io.jsonwebtoken.ExpiredJwtException;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;

/**
 * token认证工具包
 * @author huangan
 * @date 2020/11/15 14:32
 */
public class TokenUtil {
    // 15 * 60 * 1000;
    private static final long EXPIRE_TIME = 30 * 60 * 1000;
    private static final String TOKEN_SECRET = "thefirsttoken123";

    /**
     * 生成签名,15分钟过期
     * @param **username**
     * @param **password**
     * @return
     */
    public static String sign(String username, String password) {
        try {
            // 设置过期时间
            Date date = new Date(System.currentTimeMillis() + EXPIRE_TIME);
            // 私钥和加密算法
            Algorithm algorithm = Algorithm.HMAC256(TOKEN_SECRET);
            // 设置头部信息
            Map<String, Object> header = new HashMap<>(2);
            header.put("Type", "Jwt");
            header.put("alg", "HS256");
            // 返回token字符串
            return JWT.create()
                    .withHeader(header)
                    .withClaim("loginName", username)
                    .withClaim("pwd", password)
                    .withExpiresAt(date)
                    .sign(algorithm);
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    }

    /**
     * 检验token是否正确
     * @param **token**
     * @return
     */
    public static boolean verify(String token){
        try {
            Algorithm algorithm = Algorithm.HMAC256(TOKEN_SECRET);
            JWTVerifier verifier = JWT.require(algorithm).build();
            verifier.verify(token);
            return true;
        } catch (Exception e){
            return false;
        }
    }
    
    /**
     * 验证token是否失效
     * true:过期   fals
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值