今天就跟大家聊聊有關SpringBoot項目可以使用redis數據庫實現Session共享功能嗎,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內容,希望大家根據這篇文章可以有所收獲。
有時我們可能有多個不同的Web應用,可以相互調用,這時如果每個應用都有自己的session,那用戶跳轉到另一個應用時就又需要登陸一次,這樣會帶來很不好的體驗,因此我們需要在不同的應用中共享session。這里,我們采用redis來實現。
前置說明
由于只用到redis和springboot的整合,所以只能實現一個URL下的不同端口的應用之間的session共享,如果連應用名稱都完全不同的兩個應用要實現session共享,在這個基礎上還需要使用到Nginx,這種方式我暫時還沒有試過。(SpringBoot項目默認就是不帶應用名稱的,除非自己在配置文件中修改過)
需要提前在本地安裝好redis,或者連接遠程redis服務器。這里就不寫安裝教程了,可以自行去網上搜索。
添加依賴
需要為springboot項目添加以下兩個依賴,參與session共享的項目都需要添加。
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> <dependency> <groupId>org.springframework.session</groupId> <artifactId>spring-session-data-redis</artifactId> </dependency>
一個是redis的依賴,一個是spring-session-data-redis的依賴。
配置redis參數
在SpringBoot項目的application.properties配置文件中配置redis參數:
# Redis數據庫索引(默認為0) spring.redis.database=0 # Redis服務器地址,如果是遠程redis服務器,就改成服務器地址 spring.redis.host=127.0.0.1 # Redis服務器連接端口,默認是6379 spring.redis.port=6379 # 連接池最大連接數(使用負值表示沒有限制) spring.redis.lettuce.pool.max-active=8 # 連接池最大阻塞等待時間(使用負值表示沒有限制) spring.redis.lettuce.pool.max-wait=-1ms # 連接池中的最大空閑連接 spring.redis.lettuce.pool.max-idle=5 # 連接池中的最小空閑連接 spring.redis.lettuce.pool.min-idle=0 # 連接超時時間(毫秒) spring.redis.timeout=5000 spring.session.store-type=redis
如果你的項目使用的是application.yml,就進行如下配置:
spring:
 redis:
  database: 0
  host: 127.0.0.1
  port: 6379
  lettuce:
   pool:
    max-idle: 8
    min-idle: 0
    max-active: 8
    max-wait: -1ms
  timeout: 5000
 session:
  store-type: redis配置session過期時間
創建一個用于配置session過期時間的配置類:
import org.springframework.context.annotation.Configuration;
import org.springframework.session.data.redis.config.annotation.web.http.EnableRedisHttpSession;
@Configuration
@EnableRedisHttpSession(maxInactiveIntervalInSeconds = 86400*30)
public class SessionConfig {
}簡單的登錄邏輯
 @RequestMapping("/doLogin")
 public String doLogin(HttpServletRequest request, Model model){
   String username = request.getParameter("username");
   String password = request.getParameter("password");
   if(StringUtils.isEmpty(username) || StringUtils.isEmpty(password)){
     model.addAttribute("errorMsg", "用戶名和密碼不能為空");
     return "login";
   }
   // 查找該用戶,成功后根據該用戶的類別返回到對應頁面
   User user = userService.getUserByUsernameAndPassword(username, password);
   if(user == null) {
     model.addAttribute("errorMsg", "用戶名或密碼錯誤");
     return "login";
   } else {
     request.getSession().setAttribute("currentUser", user);
     model.addAttribute("currentUser", user);
     String identity = user.getIdentity();
     if("admin".equals(identity)){
       return "admin";
     }else{
       return "user";
     }
   }
 }直接按照原來的方式將對象存入session:request.getSession().setAttribute("currentUser", user); 此時session會存入redis。
登錄過濾器
@Component
public class LoginInterceptor implements HandlerInterceptor {
  @Override
  public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
      throws Exception {
    HttpSession session = request.getSession();
    User currentUser = (User) session.getAttribute("currentUser");
    if(currentUser == null){
      response.sendRedirect(request.getContextPath() + "/toLogin");
      return false;
    }else{
      return true;
    }
  }
  @Override
  public void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse,
              Object o, ModelAndView modelAndView) throws Exception {
  }
  @Override
  public void afterCompletion(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse,
                Object o, Exception e) throws Exception {
  }
}同樣按原來的方式從session中取出對象:User currentUser = (User) session.getAttribute("currentUser"); 此時會從redis中取出該對象。
注意
如果只是存字符串等redis可以直接解析的對象,那就不會有什么問題,但是如果是存取對象就需要進行序列化了,比如上文中存的是我自定義的一個User對象,那么在存的時候,是會對該對象進行序列化的,取出時也會進行反序列化,因此該對象要實現Serializable接口,并且需要進行session共享的項目中都要有一個一模一樣的對象,比如我的User定義如下:
import java.io.Serializable;
public class User implements Serializable {
  private String id;
  private String username;
  private String password;
  private String email;
  private String identity;
  private static final long serialVersionUID = -5809782578272943999L;
  
  // 省略getter、setter方法
}注意這個序列號serialVersionUID,不同應用中的User對象的這個序列號必須相同,否則無法正確進行反序列化。
看完上述內容,你們對SpringBoot項目可以使用redis數據庫實現Session共享功能嗎有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。