基于SpringBoot+Redis的Session共享與單點(diǎn)登錄詳解
前言
使用Redis來(lái)實(shí)現(xiàn)Session共享,其實(shí)網(wǎng)上已經(jīng)有很多例子了,這是確保在集群部署中最典型的redis使用場(chǎng)景。在SpringBoot項(xiàng)目中,其實(shí)可以一行運(yùn)行代碼都不用寫,只需要簡(jiǎn)單添加添加依賴和一行注解就可以實(shí)現(xiàn)(當(dāng)然配置信息還是需要的)。
然后簡(jiǎn)單地把該項(xiàng)目部署到不同的tomcat下,比如不同的端口(A、B),但項(xiàng)目訪問路徑是相同的。此時(shí)在A中使用set方法,然后在B中使用get方法,就可以發(fā)現(xiàn)B中可以獲取A中設(shè)置的內(nèi)容。
但如果就把這樣的一個(gè)項(xiàng)目在多個(gè)tomcat中的部署說(shuō)實(shí)現(xiàn)了單點(diǎn)登錄,那就不對(duì)了。
所謂單點(diǎn)登錄是指在不同的項(xiàng)目中,只需要任何一個(gè)項(xiàng)目登錄了,其他項(xiàng)目不需要登錄。
同樣是上面的例子,我們把set和get兩個(gè)方法分別放到兩個(gè)項(xiàng)目(set、get)中,并且以集群方式把兩個(gè)項(xiàng)目都部署到服務(wù)器A和B中,然后分別訪問A服務(wù)器的set和B服務(wù)器的get,你就會(huì)發(fā)現(xiàn)完全得不到你想要的結(jié)果。
同一項(xiàng)目中的set/get
依賴添加就不說(shuō)了,直接使用最簡(jiǎn)單的方式
@SpringBootApplication
@EnableRedisHttpSession
@RestController
public class SessionShareApplication {
public static void main(String[] args) {
SpringApplication.run(SessionShareApplication.class, args);
}
@Autowired
HttpSession session;
@Autowired
HttpServletRequest req;
@GetMapping("/set")
public Object set() {
session.setAttribute("state", "state was setted.");
Map<String, Object> map = new TreeMap<>();
map.put("msg", session.getAttribute("state"));
map.put("serverPort", req.getLocalPort());
return map;
}
@GetMapping("/get")
public Object get() {
Map<String, Object> map = new TreeMap<>();
map.put("msg", session.getAttribute("state"));
map.put("serverPort", req.getLocalPort());
return map;
}
}
將該項(xiàng)目打war包,分別部署在tomcatA(端口8080),tomcatB(端口8081),然后通過(guò)tomcatA/set 方法設(shè)置session,再使用 tomcatB/get 方法即可獲得session的值。但這只是實(shí)現(xiàn)了同一項(xiàng)目session的共享。并不是單點(diǎn)登錄。
為了驗(yàn)證,我們不仿將set/get方法拆分為兩個(gè)項(xiàng)目。
拆分set/get為兩個(gè)項(xiàng)目
get項(xiàng)目
@SpringBootApplication
@EnableRedisHttpSession
@RestController
public class SetApplication {
public static void main(String[] args) {
SpringApplication.run(SetApplication.class, args);
}
@Autowired
HttpSession session;
@Autowired
HttpServletRequest req;
@GetMapping("/")
public Object set() {
session.setAttribute("state", "state was setted.");
Map<String, Object> map = new TreeMap<>();
map.put("msg", session.getAttribute("state"));
map.put("serverPort", req.getLocalPort());
return map;
}
}
將該項(xiàng)目打包為set.war
set項(xiàng)目
@SpringBootApplication
@EnableRedisHttpSession
@RestController
public class GetApplication {
public static void main(String[] args) {
SpringApplication.run(GetApplication.class, args);
}
@Autowired
HttpSession session;
@Autowired
HttpServletRequest req;
@GetMapping("/")
public Object get() {
Map<String, Object> map = new TreeMap<>();
map.put("msg", session.getAttribute("state"));
map.put("serverPort", req.getLocalPort());
return map;
}
}
將該項(xiàng)目打包為get.war
再分別將set.war,get.war部署在tomcatA和tomcatB,再通過(guò) tomcatA/set 設(shè)置session內(nèi)容, 然后通過(guò) tomcatB/get 就發(fā)現(xiàn)無(wú)法獲得session的值。
問題分析
盡管我們使用的路徑都是一樣的,但其實(shí)是兩個(gè)項(xiàng)目,與前面的一個(gè)項(xiàng)目是完全不同的,問題就在于 session和cookie在默認(rèn)情況下是與項(xiàng)目路徑相關(guān)的,在同一個(gè)項(xiàng)目的情況下兩個(gè)方法所需要的cookie依賴的項(xiàng)目路徑是相同的,所以獲取session的值就沒有問題,但在后一種情況下,cookie的路徑是分別屬于不同的項(xiàng)目的,所以第二個(gè)項(xiàng)目就無(wú)法獲得第一個(gè)項(xiàng)目中設(shè)置的session內(nèi)容了。
解決方法
解決方法在springboot項(xiàng)目中其實(shí)也非常簡(jiǎn)單。既然cookie路徑發(fā)生了變化,那我們讓它配置為相同的路徑就解決了。
在每個(gè)子項(xiàng)目中都添加一個(gè)配置類或者直接設(shè)置cookie的路徑,如果有域名還可以設(shè)置域名的限制,比如 set.xxx.com 與 get.xxx.com 這種情況與我們就需要設(shè)置cookie的域名為 xxx.com,以確保無(wú)法在哪個(gè)項(xiàng)目下都能夠獲取 xxx.com 這個(gè)域名下的cookie值。這樣就確保能夠正常獲得共享的session值了。
@Configuration
public class CookieConfig {
@Bean
public static DefaultCookieSerializer defaultCookieSerializer() {
DefaultCookieSerializer serializer = new DefaultCookieSerializer();
serializer.setCookiePath("/");
//serializer.setDomainName("xxx.com"); //如果使用域名訪問,建議對(duì)這一句進(jìn)行設(shè)置
return serializer;
}
}
以上才是正直的redis實(shí)現(xiàn)單點(diǎn)登錄的正確打開方式。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
AJAX中Get請(qǐng)求報(bào)錯(cuò)404的原因以及解決辦法
剛學(xué)習(xí)一門技術(shù)時(shí)總會(huì)踩一些坑,下面這篇文章主要給大家介紹了關(guān)于AJAX中Get請(qǐng)求報(bào)錯(cuò)404的原因及解決辦法的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-03-03
SpringBoot整合MongoDB實(shí)現(xiàn)事務(wù)管理
Spring Boot是一種快速開發(fā)Spring應(yīng)用的方式,它提供了大量的自動(dòng)配置和默認(rèn)設(shè)置,以簡(jiǎn)化開發(fā)流程,MongoDB是一個(gè)基于文檔的NoSQL數(shù)據(jù)庫(kù),本文將介紹如何在Spring Boot應(yīng)用中整合MongoDB,并實(shí)現(xiàn)事務(wù)管理,需要的朋友可以參考下2024-07-07
IDEA插件Statistic統(tǒng)計(jì)代碼快速分辨爛項(xiàng)目
這篇文章主要為大家介紹了使用IDEA插件Statistic來(lái)統(tǒng)計(jì)項(xiàng)目代碼,幫助大家快速識(shí)別出爛項(xiàng)目,有需要的朋友可以借鑒參考下,希望能夠有所幫助2022-01-01
Java 如何使用Feign發(fā)送HTTP請(qǐng)求
這篇文章主要介紹了Java 如何使用Feign發(fā)送HTTP請(qǐng)求,幫助大家更好的理解和學(xué)習(xí)Java,感興趣的朋友可以了解下2020-11-11
springboot下使用shiro自定義filter的個(gè)人經(jīng)驗(yàn)分享
這篇文章主要介紹了springboot下使用shiro自定義filter的個(gè)人經(jīng)驗(yàn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
利用Java搭建個(gè)簡(jiǎn)單的Netty通信實(shí)例教程
這篇文章主要給大家介紹了關(guān)于如何利用Java搭建個(gè)簡(jiǎn)單的Netty通信,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
SpringBoot部署到騰訊云的實(shí)現(xiàn)示例
記錄一下自己第一次部署springboot項(xiàng)目,本文主要介紹了SpringBoot部署到騰訊云的實(shí)現(xiàn)示例,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-08-08
SpringCloud GateWay動(dòng)態(tài)路由用法
網(wǎng)關(guān)作為所有項(xiàng)目的入口,不希望重啟,因此動(dòng)態(tài)路由是必須的,動(dòng)態(tài)路由主要通過(guò)RouteDefinitionRepository接口實(shí)現(xiàn),其默認(rèn)的實(shí)現(xiàn)是InMemoryRouteDefinitionRepository,即在內(nèi)存中存儲(chǔ)路由配置,可基于這個(gè)map對(duì)象操作,動(dòng)態(tài)路由的實(shí)現(xiàn)方案有兩種2024-10-10
Java基礎(chǔ)學(xué)習(xí)之實(shí)參和形參
這篇文章主要介紹了Java基礎(chǔ)學(xué)習(xí)之實(shí)參形參,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java基礎(chǔ)的小伙伴們有一定的幫助,需要的朋友可以參考下2021-05-05
從內(nèi)存模型中了解Java final的全部細(xì)節(jié)
關(guān)于final關(guān)鍵字,它也是我們一個(gè)經(jīng)常用的關(guān)鍵字,可以修飾在類上、或者修飾在變量、方法上,以此看來(lái)定義它的一些不可變性!像我們經(jīng)常使用的String類中,它便是final來(lái)修飾的類,并且它的字符數(shù)組也是被final所修飾的。但是一些final的一些細(xì)節(jié)你真的了解過(guò)嗎2022-03-03

