jwt 學習筆記

来源:https://www.cnblogs.com/Yee-Q/archive/2023/03/25/17255645.html
-Advertisement-
Play Games

概述 JWT,Java Web Token,通過 JSON 形式作為 Web 應用中的令牌,用於在各方之間安全地將信息作為 JSON 對象傳輸,在數據傳輸過程中還可以完成數據加密、簽名等相關處理 JWT 的作用如下: 授權:一旦用戶登錄,每個後續請求將包括 JWT,從而允許用戶訪問該令牌允許的路由, ...



概述

JWT,Java Web Token,通過 JSON 形式作為 Web 應用中的令牌,用於在各方之間安全地將信息作為 JSON 對象傳輸,在數據傳輸過程中還可以完成數據加密、簽名等相關處理

JWT 的作用如下:

  • 授權:一旦用戶登錄,每個後續請求將包括 JWT,從而允許用戶訪問該令牌允許的路由,服務和資源
  • 信息交換:JSON Web Token 是在各方之間安全地傳輸信息的好方法,因為可以對 JWT 進行簽名,此外,由於簽名是使用標頭和有效負載計算的,因此還可以驗證內容是否篡改

傳統的 Session 認證

1. 認證方式

http 協議本身是一種無狀態協議,這就意味著如果用戶向我們的應用提供了用戶名和密碼進行認證,那麼下次請求時還需再作一次認證。因為根據 http 協議,我們並不知道是哪個用戶發出的請求,所以為了讓應用能識別是哪個用戶發出的請求,我們只能在服務存儲一份用戶登錄的信息,這份登錄信息會在響應傳遞給瀏覽器,告訴其保存為 cookie,以便下次請求時發送回來,這樣我們就能識別請求來自哪個用戶了

2. 缺點

  • 每個用戶經過認證後,都要在服務端做一次記錄,以方便下次鑒別。通常而言,session 都是保存在記憶體中的,隨著認證用戶的增多,服務端的開銷也會明顯增大
  • 用戶認證之後,服務端做認證記錄,如果認證的記錄被保存到記憶體中,就意味著下次用戶請求還得訪問該伺服器,才能拿到授權的資源,在分散式應用上,相應的限制了負載均衡器的能力,也就意味著限制了應用的擴展能力
  • 基於 cookie 進行用戶識別,cookie 一旦被捕獲,用戶就會很容易受到跨站請求偽造的攻擊
  • 在前後端分離時增加了部署的複雜性

JWT 認證

1. 認證方式

  • 前端通過 Web 表單將自己的用戶名和密碼發送給後端的介面,這一過程一般是 http post 請求,建議的方式是通過 SSL 加密的傳輸(https),從而避免敏感信息被嗅探
  • 後端核對用戶名和密碼成功後,將用戶的 id 等其他信息作為 JWT Payload(負載),將其與頭部分別進行 Base64 編碼,拼接後簽名,形成一個 JWT Token
  • 後端將 JWT 字元串作為登錄成功的返回結果返回,前端可以將返回的結果保存在本地緩存上,退出登錄時前端刪除保存的 JWT 即可
  • 前端在每次請求後端帶回 JWT
  • 後端檢查是否存在,如驗證 JWT 的有效性,檢查簽名是否正確,檢查 Token 是否過期,檢查 Token 接收方是否是自己
  • 驗證通過後,後端使用 JWT 中包含的用戶信息進行其他邏輯操作,返回相應結果

2. 優點

  • 簡潔,可以通過 URL、POST 參數或者在 HTTP Header 發送,因為數據量小,傳輸速度快
  • 自包含,負載中包含了所有用戶需要的信息,避免多次查詢資料庫
  • Token 以加密形式保存在客戶端,原則上任何 web 形式都支持
  • 不需要在服務端保存會話信息,特別適用於分散式微服務

JWT 結構

通常 JWT 如下所示:xxxxx.yyyyy.zzzzz

  • 標頭(Header):標頭一般由兩部分組成:令牌的類型和所使用的簽名演算法,標頭使用 Base64 編碼

    {
        "alg":"HS256",
        "typ":"JWT"
    }
    
  • 有效負載(payload):令牌的第二部分是有效負載,其中包含聲明。聲明是有關實體(通常是用戶)和其他數據的聲明。同樣使用 Base64 編碼。不建議放入用戶的敏感信息

    {
        "sub":"12345678",
        "name":"john",
        "admin":true,
        ...
    }
    
  • 簽名(Signature):Signature 需要使用編碼後的 Header 和 Payload 以及我們提供的一個密鑰,然後使用 Header 中指定簽名演算法,進行簽名。簽名的作用是保證 JWT 沒有被篡改過,如:HMACSHA256(base64UrlEncode(header) + "." + base64UrlEncode(payload),secret)

客戶端收到服務端發送的 token 後,再次請求服務端需要帶上 token,此時 token 包含三部分:經過 Base64 編碼的 Header、經過 Base64 編碼的 Payload 和加密後的簽名,服務端用自己保存的 secret 與客戶端發送的 Header、Payload 運算,如果結果和客戶端帶回來的簽名不一致,則驗證失敗


JWT 使用

引入依賴

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

生成 token

Calendar instance = Calendar.getInstance();
instance.add(Calendar.SECOND, 20);
Map<String, Object> map = new HashMap<>();
String token = JWT.create().withHeader(map)    // header
    .withClaim("userId", 21)    // payload
    .withClaim("username", "yeeq")
    .withExpiresAt(instance.getTime())    // 指定令牌的過期時間
    .sign(Algorithm.HMAC256("FAWF2#!F@"));  // 簽名,並指定密鑰

根據令牌和簽名解析數據

// 創建驗證對
JWTVerifier jwtVerifier = JWT.require(Algorithm.HMAC256("FAWF2#!F@")).build();
// 解碼後的信息
DecodedJWT verify = jwtVerifier.verify("eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJleHAiOjE2MDAyNTQ4NzIsInVzZXJJZCI6MjEsInVzZXJuYW1lIjoieWVlcSJ9.jo_6gKThSXUcEfH1e9bu7at9lm2zmdupwiYvMUWopls");
System.out.println(verify.getClaim("userId").asInt());
System.out.println(verify.getClaim("username").asString());

常見異常信息:

  • SignatureVerificationException:簽名不一致異常
  • TokenExpiredException:令牌過期異常
  • AlgorithmMismatchException:演算法不匹配異常
  • InvalidClaimException:失效的 payload 異常

JWT 封裝工具類

一般結合攔截器或者網關完成認證

public class JWTUtils {

    private static final String SIGN = "FAWF2#!F@";

    /**
         * 生成 token
         * @param map payload 的信息
         * @return token
         */
    public static String getToken(Map<String, String> map) {
        Calendar instance = Calendar.getInstance();
        instance.add(Calendar.DATE, 7);
        JWTCreator.Builder builder = JWT.create();
        // 創建 payload
        map.forEach((k, v) -> {
            builder.withClaim(k, v);
        });
        String token = builder.withExpiresAt(instance.getTime())    // 指定令牌的過期時間
            .sign(Algorithm.HMAC256(SIGN));  // 簽名,並指定密鑰
        return token;
    }

    /**
         * 驗證 token 的合法性
         * @param token token
         */
    public static void verifyJWT(String token) {
        JWT.require(Algorithm.HMAC256(SIGN)).build().verify(token);
    }

    /**
         * 獲取 token 的信息
         * @return token 的信息
         */
    public static DecodedJWT getTokenInfo(String token) {
        DecodedJWT verify = JWT.require(Algorithm.HMAC256(SIGN)).build().verify(token);
        return verify;
    }
}


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 代理模式(Proxy Pattern)是一種結構型設計模式,結構型模式描述如何將類或對象按某種佈局組成更大的結構。它允許你提供一個代理對象來控制對另一個對象的訪問。代理對象擁有與實際對象相同的介面,因此它可以被用來代替實際對象。 ...
  • 在軟體行業,對於什麼是架構,都有很多的爭論,每個人都有自己的理解。在不同的書籍上, 不同的作者, 對於架構的定義也不統一, 角度不同, 定義不同。此君說的架構和彼君理解的架構未必是一回事。因此我們在討論架構之前,我們先討論架構的概念定義, 因為概念是人認識這個世界的基礎和用來溝通的手段,如果對架構概... ...
  • 超級大的數做加減乘除 java有八大數據類型: 1、byte(位),最大存儲數據量是255; 2、short(短整數),最大數據存儲量是65536; 3、int(整數),最大數據存儲容量是2的32次方減1; 4、long(長整數),最大數據存儲容量是2的64次方減1; 5、float(單精度浮動數) ...
  • Java已經誕生20多年了,依然是企業級開發中使用最廣泛的語言,也是挨罵最多的語言。Java廣受批評的四個缺點是:性能差、記憶體消耗大、GUI弱、代碼啰嗦,我們應該如何看待這幾個問題呢?在微服務的背景下,提倡圍繞業務能力而非技術來構建應用,允許由不同的語言構建應用程式。一個超大的集群,往往有上萬個微服... ...
  • 什麼是Servlet? Servlet是使用Java語言編寫的運行在伺服器端的程式。狹義的Servlet是指Java語言實現的一個介面,廣義的Servlet是指任何實現了這個Servlet介面的類,一般情況下,人們將Servlet理解為後者。Servlet 主要用於處理客戶端傳來的 HTTP 請求, ...
  • 本文已經收錄到Github倉庫,該倉庫包含電腦基礎、Java基礎、多線程、JVM、資料庫、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分散式、微服務、設計模式、架構、校招社招分享等核心知識點,歡迎star~ Github地址 Spring的優點 通過控制反轉 ...
  • 電腦編程說到底還是程式員的思維體現,人情世故也會反映在代碼的邏輯上, 寫代碼就如學做人一樣,從哪裡來到哪裡去 ...
  • SpringBoot整合MyBatis/Plus 1.SpringBoot整合MyBatis 1.1整合案例 需求:整合SpringBoot和Mybatis,向資料庫中查詢數據。 項目結構: 1.1.1創建資料庫和表 -- 創建資料庫 DROP DATABASE IF EXISTS springbo ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...