主頁 > 後端開發 > 使用 Sa-Token 實作不同的登錄模式:單地登錄、多地登錄、同端互斥登錄

使用 Sa-Token 實作不同的登錄模式:單地登錄、多地登錄、同端互斥登錄

2023-07-10 07:46:29 後端開發

一、需求分析

如果你經常使用騰訊QQ,就會發現它的登錄有如下特點:它可以手機電腦同時在線,但是不能在兩個手機上同時登錄一個賬號,

同端互斥登錄,指的就是:像騰訊QQ一樣,在同一型別設備上只允許單地點登錄,在不同型別設備上允許同時在線,

動態演示圖:

同端互斥登錄

Sa-Token 是一個輕量級 java 權限認證框架,主要解決登錄認證、權限認證、單點登錄、OAuth2、微服務網關鑒權 等一系列權限相關問題,
Gitee 開源地址:https://gitee.com/dromara/sa-token

本文將介紹在 Sa-Token 中,如何實作以下登錄策略:

  • 單地登錄:指一個賬號同一時間只能在一個地方登錄,新登錄會擠掉舊登錄,也可以叫:單端登錄,
  • 多地登錄:指一個賬號同一時間可以在不同地方登錄,新登錄會和舊登錄共存,也可以叫:多端登錄,
  • 同端互斥登錄:在同一型別設備上只允許單地點登錄,在不同型別設備上允許同時在線,參考騰訊QQ的登錄模式:手機和電腦可以同時在線,但不能兩個手機同時在線,

與之對應的,注銷策略也將分為以下幾種:

  • 單端注銷:只在呼叫退出的一端注銷,
  • 全端注銷:一端注銷,全端下線,
  • 同端注銷:例如將所有手機端注銷下線,PC端不受影響,

二、多地登錄

此模式較為簡單,Sa-Token 默認模式即為多地登錄模式,

1、首先引入 Sa-Token 依賴:
<!-- Sa-Token 權限認證 -->
<dependency>
	<groupId>cn.dev33</groupId>
	<artifactId>sa-token-spring-boot-starter</artifactId>
	<version>1.34.0</version>
</dependency>

注:如果你使用的是 SpringBoot 3.x,只需要將 sa-token-spring-boot-starter 修改為 sa-token-spring-boot3-starter 即可,

2、在用戶登錄時將賬號id寫入會話中
@RestController
@RequestMapping("/user/")
public class UserController {
	@RequestMapping("doLogin")
	public SaResult doLogin(String username, String password) {
		// 此處僅作示例模擬,真實專案需要從資料庫中查詢資料進行比對 
		if("zhang".equals(username) && "123456".equals(password)) {
			StpUtil.login(10001);
			return SaResult.ok("登錄成功");
		}
		return SaResult.ok("登錄失敗");
	}
}

啟動類:

@SpringBootApplication
public class SaTokenDemoApplication {
	public static void main(String[] args) {
		SpringApplication.run(SaTokenDemoApplication.class, args); 
		System.out.println("\n啟動成功:Sa-Roken 配置如下:" + SaManager.getConfig());
	}
}

如上代碼,在多人登錄同一賬號時將不會對舊會話做任何處理,同一賬號可以在多個地點任意登錄,互不影響,

3、如果要全端注銷,可以呼叫 logout 方法:
// 會話注銷
@RequestMapping("logout")
public SaResult logout() {
	StpUtil.logout();
	return SaResult.ok("退出登錄成功");
}

呼叫如上方法注銷后,當前賬號所有端將一起下線,

4、單端注銷

如果要只注銷一端,可將組態檔中 is-share 的值配置為 false

sa-token:
	is-share: false

此配置項的含義為:在多人登錄同一賬號時,是否共用一個 Token,

  • 為 true 時:所有登錄共用一個 Token,
  • 為 false 時:每次登錄新建一個 Token,

此值為 false 后,每次登錄都將回傳不同的 Token,與之對應的,呼叫 StpUtil.logout() 也只會注銷掉當前的 Token,其他端不受影響,

三、單地登錄

單地登錄的重點是需要改一下 yml 組態檔:

sa-token: 
	is-concurrent: false

is-concurrent 的含義為是否允許同一賬號并發登錄:

  • 為 true 時:允許一起登錄,
  • 為 false 時:新登錄擠掉舊登錄,

其它代碼與 [多地登錄] 無異,當我們在兩個瀏覽器分別登錄同一賬號時,舊會話再次訪問系統將會得到如下提示:

{
	"code": 401,
	"msg": "Token 已被頂下線",
	"data": null
}

在 單地登錄 模式中,不存在注銷策略的問題,因為同一時間內,一個賬號最多在一個設備在線,只要呼叫注銷,就必然是全端下線,

四、同端互斥登錄

好了,終于輪到主角出場,同端互斥登錄可以讓我們像騰訊QQ一樣,在同一型別設備上只允許單地點登錄,在不同型別設備上允許同時在線,

那么在 Sa-Token 中如何做到同端互斥登錄呢?

首先如 單地登錄一樣,在組態檔中,將 sa-token.is-concurrent 配置為false,然后呼叫登錄等相關介面時宣告設備標識即可:

1、指定設備標識登錄
StpUtil.login(10001, "PC");    

呼叫此方法登錄后,同設備的會被頂下線(不同設備不受影響),再次訪問系統時會拋出 NotLoginException 例外,場景值=-4

場景值 對應常量 含義說明
-1 NotLoginException.NOT_TOKEN 未能從請求中讀取到 Token
-2 NotLoginException.INVALID_TOKEN 已讀取到 Token,但是 Token無效
-3 NotLoginException.TOKEN_TIMEOUT 已讀取到 Token,但是 Token已經過期
-4 NotLoginException.BE_REPLACED 已讀取到 Token,但是 Token 已被頂下線
-5 NotLoginException.KICK_OUT 已讀取到 Token,但是 Token 已被踢下線

如果第二個引數填寫null或不填,代表將這個賬號id所有在線端踢下線,被踢出者再次訪問系統時會拋出 NotLoginException 例外,場景值=-5

2、查詢當前登錄的設備標識
StpUtil.getLoginDevice(); 

如果在登錄時未指定設備型別值,呼叫此方法將回傳默認值:default-device

3、指定設備端型別下線

業務場景舉例:在手機端控制PC端下線(手機端本身不受影響)

StpUtil.logout(10001, "PC");		
4、全端下線

在呼叫 logout 方法時,不填寫具體的設備端型別,將默認控制所有端一起下線,

StpUtil.logout(10001);		

以上就是 Sa-Token 框架在處理登錄問題時的各種方案,可以看出不管是簡單的多地登錄還是復雜的同端互斥登錄,在 Sa-Token 都有完善的解決方案,


參考資料

  • Sa-Token 檔案:https://sa-token.cc
  • Gitee 倉庫地址:https://gitee.com/dromara/sa-token
  • GitHub 倉庫地址:https://github.com/dromara/sa-token

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/556870.html

標籤:其他

上一篇:Java擴展Nginx之二:編譯nginx-clojure原始碼

下一篇:返回列表

標籤雲
其他(162274) Python(38272) JavaScript(25528) Java(18293) C(15239) 區塊鏈(8275) C#(7972) AI(7469) 爪哇(7425) MySQL(7292) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5876) 数组(5741) R(5409) Linux(5347) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4615) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2438) ASP.NET(2404) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) HtmlCss(1993) .NET技术(1986) 功能(1967) Web開發(1951) C++(1942) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1882) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • 使用 Sa-Token 實作不同的登錄模式:單地登錄、多地登錄、同端互斥

    ### 一、需求分析 如果你經常使用騰訊QQ,就會發現它的登錄有如下特點:它可以手機電腦同時在線,但是不能在兩個手機上同時登錄一個賬號。 同端互斥登錄,指的就是:像騰訊QQ一樣,在同一型別設備上只允許單地點登錄,在不同型別設備上允許同時在線。 動態演示圖: ![同端互斥登錄](https://oss ......

    uj5u.com 2023-07-10 07:46:29 more
  • Java擴展Nginx之二:編譯nginx-clojure原始碼

    ### 歡迎訪問我的GitHub > 這里分類和匯總了欣宸的全部原創(含配套原始碼):[https://github.com/zq2599/blog_demos](https://github.com/zq2599/blog_demos) ### 為什么要編譯nginx-clojure原始碼 - 作為《 ......

    uj5u.com 2023-07-10 07:41:03 more
  • Spring 中 Bean 的配置細節

    大家好,我是 god23bin,今天繼續說 Spring 的內容,關于 Spring 中 Bean 的配置的,通過上一篇文章的學習,我們知道了 Spring 中的依賴注入,其中有兩種主要的方式,分別是基于構造方法的 DI 和 基于 Setter 的 DI。 ......

    uj5u.com 2023-07-10 07:40:54 more
  • 暗黑2能用Java開發?還能生成APP?

    > 最近燒哥發現個寶藏專案,竟然用Java開發了暗黑2出來。 眾所周知,暗黑2是暴雪開發的一款經典游戲,距今雖有20多年,仍然有很多粉絲。 粉絲延續熱情的方式有很多,一種是做Mod,比如[魔電](https://www.median-xl.com/),對怪物、技能、物品、場景、甚至游戲機制都有大改, ......

    uj5u.com 2023-07-10 07:39:57 more
  • Object.equals 和 String.equals的區別

    一. 原始碼展示: 1. Object.equals: ①參考型別地址值比較,直接回傳結果:true || false public class Object { public boolean equals(Object obj) { return (this == obj); } } 2. Stri ......

    uj5u.com 2023-07-09 07:49:47 more
  • python筆記:第四章使用字典

    ## 1.1 概述 > 說白了就是鍵值對的映射關系 > > 不會丟失資料本身關聯的結構,但不關注資料的順序 > > 是一種可變型別 ```py 格式:dic = {鍵:值, 鍵:值} ``` * 鍵的型別:字典的鍵可以是任何不可變的型別,如浮點數,字串,元組 ## 1.2 函式dict 可以從其他 ......

    uj5u.com 2023-07-09 07:49:44 more
  • Java中AQS的原理與實作

    ### 目錄 *1:什么是AQS?* *2:AQS都有那些用途?* *3:我們如何使用AQS* *4:AQS的實作原理* *5:對AQS的設計與實作的一些思考* ### 1:什么是AQS ? 隨著計算機的算力越來越強大,各種各樣的并行編程模型也隨即踴躍而來,但當我們要在并行計算中使用共享資源的時候, ......

    uj5u.com 2023-07-09 07:49:30 more
  • 計算機COM口資料測驗

    計算機COM口資料測驗一、基本使用流程 程式需要以管理員身份運行,COM口回路測驗需短接2,3pin,測驗時候使用控制臺,配置測驗相關路徑,并在測驗完成后 1.測驗配置路徑D:\bigdata\INI\FWCOM.ini 2.測驗完成后需要在路徑D:\bigdata\LOG\生成測驗FWCOM.lo ......

    uj5u.com 2023-07-09 07:49:18 more
  • Python潮流周刊#10:Twitter 的強敵 Threads 是用 Python 開發的!

    你好,我是貓哥。這里每周分享優質的 Python 及通用技術內容,大部分為英文,已在小標題注明。(標題取自其中一則分享,不代表全部內容都是該主題,特此宣告。) 首發于我的博客:[https://pythoncat.top/posts/2023-07-08-weekly](https://python ......

    uj5u.com 2023-07-09 07:48:59 more
  • C++ 慣用法之 Copy-Swap 拷貝交換

    # C++ 慣用法之 Copy-Swap 拷貝交換 > 這是“C++ 慣用法”合集的第 3 篇,前面 2 篇分別介紹了 RAII 和 PIMPL 兩種慣用法: > > - [RAII: Resouce Acquistion Is Initialization](https://www.cnblogs ......

    uj5u.com 2023-07-09 07:48:48 more