博客 / 詳情

返回

一文詳解 Sa-Token 中的 SaSession 對象

Sa-Token 是一個輕量級 java 權限認證框架,主要解決登錄認證、權限認證、單點登錄、OAuth2、微服務網關鑑權 等一系列權限相關問題。

Gitee 開源地址:https://gitee.com/dromara/sa-token

本文將詳細介紹 Sa-Token 中的不同 SaSession 對象的區別,以及各種方便的存取值的方法。

一、Session 是什麼?

Session 是會話中專業的數據緩存組件,通過 Session 我們可以很方便的緩存一些高頻讀寫數據,提高程序性能,例如:

// 在登錄時緩存user對象 
StpUtil.getSession().set("user", user);

// 然後我們就可以在任意處使用這個user對象
SysUser user = (SysUser) StpUtil.getSession().get("user");

在 Sa-Token 中,SaSession 分為三種,分別是:

  • User-Session: 指的是框架為每個 賬號id 分配的 SaSession。
  • Token-Session: 指的是框架為每個 token 分配的 SaSession。
  • Custom-Session: 指的是以一個 特定的值 作為SessionId,來分配的 Session。

假設三個客户端登錄同一賬號,且配置了不共享token,那麼此時的Session模型是:

 title=

簡而言之:

  • User-Session 以UserId為主,只要token指向的UserId一致,那麼對應的Session對象就一致。
  • Token-Session 以token為主,只要token不同,那麼對應的Session對象就不同。
  • Custom-Session 以特定的key為主,不同key對應不同的Session對象,同樣的key指向同一個Session對象。

二、獲取 User-Session

有關賬號Session的API如下:

// 獲取當前賬號id的Session (必須是登錄後才能調用)
StpUtil.getSession();

// 獲取當前賬號id的Session, 並決定在Session尚未創建時,是否新建並返回
StpUtil.getSession(true);

// 獲取賬號id為10001的Session
StpUtil.getSessionByLoginId(10001);

// 獲取賬號id為10001的Session, 並決定在Session尚未創建時,是否新建並返回
StpUtil.getSessionByLoginId(10001, true);

// 獲取SessionId為xxxx-xxxx的Session, 在Session尚未創建時, 返回null 
StpUtil.getSessionBySessionId("xxxx-xxxx");

三、獲取 Token-Session

有關令牌Session的API如下:

// 獲取當前 Token 的 Token-Session 對象
StpUtil.getTokenSession();

// 獲取指定 Token 的 Token-Session 對象
StpUtil.getTokenSessionByToken(token);

四、獲取自定義Session

自定義Session指的是以一個特定的值作為SessionId來分配的Session, 藉助自定義Session,你可以為系統中的任意元素分配相應的session

例如以商品id作為key為每個商品分配一個Session,以便於緩存和商品相關的數據,其相關API如下:

// 查詢指定key的Session是否存在
SaSessionCustomUtil.isExists("goods-10001");

// 獲取指定key的Session,如果沒有,則新建並返回
SaSessionCustomUtil.getSessionById("goods-10001");

// 獲取指定key的Session,如果沒有,第二個參數決定是否新建並返回  
SaSessionCustomUtil.getSessionById("goods-10001", false);   

// 刪除指定key的Session
SaSessionCustomUtil.deleteSessionById("goods-10001");

五、在 Session 上存取值

// 寫值 
session.set("name", "zhang"); 

// 寫值 (只有在此key原本無值的時候才會寫入)
session.setDefaultValue("name", "zhang");

// 取值
session.get("name");

// 取值 (指定默認值)
session.get("name", "<defaultValue>"); 

// 取值 (若無值則執行參數方法, 之後將結果保存到此鍵名下,並返回此結果   若有值則直接返回, 無需執行參數方法)
session.get("name", () -> {
            return ...;
        });

// ---------- 數據類型轉換: ----------
session.getInt("age");         // 取值 (轉int類型)
session.getLong("age");        // 取值 (轉long類型)
session.getString("name");     // 取值 (轉String類型)
session.getDouble("result");   // 取值 (轉double類型)
session.getFloat("result");    // 取值 (轉float類型)
session.getModel("key", Student.class);     // 取值 (指定轉換類型)
session.getModel("key", Student.class, <defaultValue>);  // 取值 (指定轉換類型, 並指定值為Null時返回的默認值)

// 是否含有某個key (返回true或false)
session.has("key"); 

// 刪值 
session.delete('name');          

// 清空所有值 
session.clear();                 

// 獲取此 Session 的所有key (返回Set<String>)
session.keys();      

六、其它操作

// 返回此 Session 的id 
session.getId();                          

// 返回此 Session 的創建時間 (時間戳) 
session.getCreateTime();                  

// 返回此 Session 會話上的底層數據對象(如果更新map裏的值,請調用session.update()方法避免產生髒數據)
session.getDataMap();                     

// 將這個 Session 從持久庫更新一下
session.update();                         

// 註銷此 Session 會話 (從持久庫刪除此Session)
session.logout();                         

七、SaSession 環境隔離説明

有同學經常會把 SaSessionHttpSession 進行混淆,例如:

@PostMapping("/resetPoints")
public void reset(HttpSession session) {
    // 在 HttpSession 上寫入一個值 
    session.setAttribute("name", 66);
    // 在 SaSession 進行取值
    System.out.println(StpUtil.getSession().get("name"));    // 輸出null
}

要點:

  1. SaSessionHttpSession 沒有任何關係,在HttpSession上寫入的值,在SaSession中無法取出。
  2. HttpSession並未被框架接管,在使用Sa-Token時,請在任何情況下均使用SaSession,不要使用HttpSession

八、未登錄場景下獲取 Token-Session

默認場景下,只有登錄後才能通過 StpUtil.getTokenSession() 獲取 Token-Session

如果想要在未登錄場景下獲取 Token-Session ,有兩種方法:

  • 方法一:將全局配置項 tokenSessionCheckLogin 改為 false。
  • 方法二:使用匿名 Token-Session
// 獲取當前 Token 的匿名 Token-Session (可在未登錄情況下使用的 Token-Session)
StpUtil.getAnonTokenSession();

注意點:如果前端沒有提交 Token ,或者提交的 Token 是一個無效 Token 的話,框架將不會根據此 Token 創建 Token-Session 對象,
而是隨機一個新的 Token 值來創建 Token-Session 對象,此 Token 值可以通過 StpUtil.getTokenValue() 獲取到。


參考資料

  • Sa-Token 文檔:https://sa-token.cc
  • Gitee 倉庫地址:https://gitee.com/dromara/sa-token
  • GitHub 倉庫地址:https://github.com/dromara/sa-token
user avatar codingdgsun 頭像 fulng 頭像
2 位用戶收藏了這個故事!

發佈 評論

Some HTML is okay.