Spring session實(shí)現(xiàn)共享單點(diǎn)登錄案例過程解析
一、項(xiàng)目構(gòu)建
1、案例說明
本文主要演示單點(diǎn)登錄功能,會(huì)貼出主要配置和代碼以及必要解釋,全部代碼請(qǐng)參考git地址。session共享一個(gè)基本原則是將session存儲(chǔ)在某個(gè)地方,所有的應(yīng)用都可以訪問,這里使用redis存儲(chǔ)session。當(dāng)應(yīng)用需要認(rèn)證時(shí),先從redis讀取用戶信息。
2、基本配置
1)pom.xml
<dependency> <groupId>org.springframework.session</groupId> <artifactId>spring-session-data-redis</artifactId> </dependency> <dependency> <groupId>redis.clients</groupId> <artifactId>jedis</artifactId> </dependency> <dependency> <groupId>org.springframework.session</groupId> <artifactId>spring-session</artifactId> <version>1.3.1.RELEASE</version> </dependency>
2)application.properties
server.port=8080 spring.session.store-type = redis spring.redis.host=192.168.7.151 spring.redis.port=6379
本案例的兩個(gè)應(yīng)用完全一樣,一個(gè)端口是8080,一個(gè)端口是80
3、代碼變動(dòng)
1)新增SimpleImageCode.java
public class SimpleImageCode implements Serializable{
private static final long serialVersionUID = 1L;
private String code;
private LocalDateTime expireTime;
public SimpleImageCode(String code,LocalDateTime expireTime) {
this.code = code;
this.expireTime = expireTime;
}
public String getCode() {
return code;
}
public void setCode(String code) {
this.code = code;
}
public LocalDateTime getExpireTime() {
return expireTime;
}
public void setExpireTime(LocalDateTime expireTime) {
this.expireTime = expireTime;
}
public boolean isExpried() {
return LocalDateTime.now().isAfter(expireTime);
}
}
該類與ImageCode.java基本一樣,區(qū)別1:實(shí)現(xiàn)了Serializable接口;區(qū)別2:沒有BufferedImage屬性。原因是圖形驗(yàn)證碼要放入session中,而session需要存放到redis中,所以必須實(shí)現(xiàn)序列化接口。一個(gè)類實(shí)現(xiàn)序列化接口,它里面的類屬性也要實(shí)現(xiàn)序列化接口,但是BufferedImage是jdk的類,無法實(shí)現(xiàn)序列化接口,這樣就不把它放入到redis中,在校驗(yàn)時(shí),我么只會(huì)校驗(yàn)驗(yàn)證碼和過期時(shí)間,所以不會(huì)影響。
2)修改ValidateCodeController.java
@GetMapping("/code/image")
public void createCode(HttpServletRequest request,HttpServletResponse response) throws Exception {
ImageCode imageCode = createImageCode(request);
SimpleImageCode simpleImageCode = new SimpleImageCode(imageCode.getCode(),imageCode.getExpireTime());
//request.getSession().setAttribute("imageCodeSession", imageCode);
request.getSession().setAttribute("imageCodeSession", simpleImageCode);//序列化到redis中
ImageIO.write(imageCode.getImage(), "JPEG", response.getOutputStream());
}
將SimpleImageCode放入到session中
3)修改ValidateCodeFilter.java
private void validate(HttpServletRequest request){
//ImageCode codeInSession = (ImageCode)request.getSession().getAttribute("imageCodeSession");
SimpleImageCode codeInSession = (SimpleImageCode)request.getSession().getAttribute("imageCodeSession");
String codeInRequest = request.getParameter("imageCode");
... ...//校驗(yàn)邏輯
request.getSession().removeAttribute("imageCodeSession");
}
校驗(yàn)驗(yàn)證碼前從session中取出SimpleImageCode
二、測試驗(yàn)證
1)啟動(dòng)redis、80端口應(yīng)用、8080端口應(yīng)用,查看redis信息為空,如下:

2)瀏覽器輸入:localhost:8080/index.html,跳轉(zhuǎn)登錄頁面,查看redis,如下:

3)登錄后,查看redis,如下:

4)同一個(gè)瀏覽器輸入:localhost/index.html,直接跳到index頁面,查看redis,如下:

5)點(diǎn)擊index.html中的退出連接,查看redis,如下:

6)再次訪問localhost:8080/index.html,跳轉(zhuǎn)登錄頁面,查看redis,如下:

7)再次登錄8080的應(yīng)用,查看redis,如下:

通過測試發(fā)現(xiàn)實(shí)現(xiàn)了單點(diǎn)登錄。貼出截圖只是說明session存在了redis中,并且會(huì)隨著操作變化。實(shí)際無需關(guān)心redis。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot整合Redisson的步驟(單機(jī)版)
Redisson非常適用于分布式鎖,而我們的一項(xiàng)業(yè)務(wù)需要考慮分布式鎖這個(gè)應(yīng)用場景,于是我整合它做一個(gè)初步簡單的例子(和整合redis一樣)。2021-05-05
SpringBoot+WebSocket實(shí)現(xiàn)即時(shí)通訊的方法詳解
這篇文章主要為大家詳細(xì)介紹了如何利用SpringBoot+WebSocket實(shí)現(xiàn)即時(shí)通訊功能,文中示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)或工作有一定參考價(jià)值,需要的可以參考一下2022-05-05
詳解Spring事件發(fā)布與監(jiān)聽機(jī)制
Spring提供了ApplicationContext事件機(jī)制,可以發(fā)布和監(jiān)聽事件,這個(gè)特性非常有用。Spring內(nèi)置了一些事件和監(jiān)聽器,例如在Spring容器啟動(dòng)前,Spring容器啟動(dòng)后,應(yīng)用啟動(dòng)失敗后等事件發(fā)生后,監(jiān)聽在這些事件上的監(jiān)聽器會(huì)做出相應(yīng)的響應(yīng)處理2021-06-06

