赞
踩
什么是jwt?
什么是token?
结合 JWT、Token 和 Redis 进行续期的一般步骤:
生成 JWT: 用户登录成功后,服务器生成一个 JWT,并返回给客户端。
import io.jsonwebtoken.Jwts;
import io.jsonwebtoken.SignatureAlgorithm;
public class JwtUtil {
private static final String SECRET_KEY = "yourSecretKey";
public static String generateToken(String subject) {
return Jwts.builder()
.setSubject(subject)
.signWith(SignatureAlgorithm.HS256, SECRET_KEY)
.compact();
}
}
将 JWT 存储到 Redis: 服务器将生成的 JWT 存储到 Redis 中,以便后续验证和续期。
import redis.clients.jedis.Jedis;
public class RedisUtil {
private static final String REDIS_HOST = "localhost";
private static final int REDIS_PORT = 6379;
public static void storeTokenInRedis(String key, String token) {
try (Jedis jedis = new Jedis(REDIS_HOST, REDIS_PORT)) {
jedis.set(key, token);
}
}
}
客户端保存 JWT: 客户端收到 JWT 后,将其保存在本地,通常可以存储在 Local Storage 或者 Session Storage 中。
// 将 JWT 存储到 LocalStorage
localStorage.setItem('jwtToken', token);
// 从 LocalStorage 获取 JWT
const token = localStorage.getItem('jwtToken');
// 将 JWT 存储到 Cookie
document.cookie = `jwtToken=${token}; path=/;`;
// 从 Cookie 获取 JWT
const token = document.cookie.split('; ').find(row => row.startsWith('jwtToken')).split('=')[1];
// 在 Java 中使用内存保存 JWT 的示例
String jwtToken;
使用 JWT 进行身份验证: 客户端在每次请求需要身份验证的资源时,将 JWT 放置在请求头中发送给服务器。
import io.jsonwebtoken.Claims; import io.jsonwebtoken.Jws; import io.jsonwebtoken.Jwts; public class JwtUtil { private static final String SECRET_KEY = "yourSecretKey"; public static boolean verifyToken(String token) { try { Jws<Claims> claims = Jwts.parser().setSigningKey(SECRET_KEY).parseClaimsJws(token); return true; } catch (Exception e) { return false; } } }
验证 JWT 的有效性: 服务器收到请求后,验证 JWT 的签名和有效期等信息,以确认其有效性。
下面是一个使用 Java 的示例代码,演示如何验证 JWT 的有效性:
import io.jsonwebtoken.Claims; import io.jsonwebtoken.Jwts; import io.jsonwebtoken.security.Keys; import java.security.Key; import java.util.Date; public class JwtUtil { private static final String SECRET_KEY = "yourSecretKey"; public static boolean validateJwt(String token) { try { Claims claims = Jwts.parserBuilder() .setSigningKey(Keys.hmacShaKeyFor(SECRET_KEY.getBytes())) .build() .parseClaimsJws(token) .getBody(); // 验证过期时间 Date expiration = claims.getExpiration(); if (expiration != null && expiration.before(new Date())) { return false; // JWT 已过期 } // 可选:其他声明的验证 // String username = claims.getSubject(); // List<String> roles = (List<String>) claims.get("roles"); // 进行额外的声明验证 return true; // JWT 验证通过 } catch (Exception e) { return false; // JWT 验证失败 } } }
在上述代码中,validateJwt
方法接受一个 JWT,并返回一个布尔值,指示该 JWT 是否有效。如果 JWT 有效,则返回 true
,否则返回 false
。
检查 JWT 的过期时间: 如果 JWT 即将过期,服务器可以发出一个新的 JWT,并更新 Redis 中的对应信息。
续期 JWT: 服务器根据业务逻辑和安全策略决定是否续期 JWT。如果续期,则生成一个新的 JWT,更新 Redis 中的相关信息,并将新的 JWT 返回给客户端。
public class JwtUtil {
private static final String SECRET_KEY = "yourSecretKey";
private static final long EXPIRATION_TIME = 86400000L; // 24小时
public static String refreshToken(String subject) {
return Jwts.builder()
.setSubject(subject)
.setExpiration(new Date(System.currentTimeMillis() + EXPIRATION_TIME))
.signWith(SignatureAlgorithm.HS256, SECRET_KEY)
.compact();
}
}
客户端更新 JWT: 客户端收到新的 JWT 后,将其替换掉旧的 JWT,并更新本地存储中的 Token。
import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.net.HttpURLConnection; import java.net.URL; public class UpdateJWTClient { private static String jwtToken = ""; // 存储 JWT 的变量,初始为空 public static void main(String[] args) { // 模拟从服务器获取新的 JWT String newJWT = fetchNewJWT(); if (newJWT != null && !newJWT.isEmpty()) { // 替换旧的 JWT 并更新本地存储中的 Token updateJWT(newJWT); // 使用新的 JWT 发送请求 sendRequestWithNewJWT(); } else { System.out.println("Failed to fetch new JWT."); } } // 模拟从服务器获取新的 JWT private static String fetchNewJWT() { // 这里应该发送 HTTP 请求到服务器获取新的 JWT // 这里简化为直接返回一个模拟的 JWT return "new.jwt.token.example"; } // 替换旧的 JWT 并更新本地存储中的 Token private static void updateJWT(String newToken) { jwtToken = newToken; // 替换旧的 JWT // 这里应该将新的 JWT 存储在本地,例如使用 SharedPreferences // 这里简化为打印出来 System.out.println("Updated JWT: " + jwtToken); } // 使用新的 JWT 发送请求 private static void sendRequestWithNewJWT() { try { URL url = new URL("https://example.com/api/resource"); HttpURLConnection connection = (HttpURLConnection) url.openConnection(); connection.setRequestMethod("GET"); connection.setRequestProperty("Authorization", "Bearer " + jwtToken); int responseCode = connection.getResponseCode(); if (responseCode == HttpURLConnection.HTTP_OK) { // 请求成功,读取响应 BufferedReader reader = new BufferedReader(new InputStreamReader(connection.getInputStream())); String inputLine; StringBuffer response = new StringBuffer(); while ((inputLine = reader.readLine()) != null) { response.append(inputLine); } reader.close(); System.out.println("Response: " + response.toString()); } else { // 请求失败 System.out.println("Failed to send request. Response code: " + responseCode); } } catch (IOException e) { e.printStackTrace(); } } }
在这个示例中,fetchNewJWT
方法模拟了从服务器获取新的 JWT,实际情况下应该发送 HTTP 请求到服务器获取。然后,updateJWT
方法用新的 JWT 替换旧的 JWT,并更新本地存储中的 Token。最后,sendRequestWithNewJWT
方法使用新的 JWT 发送请求到服务器。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。