1. 程式人生 > 其它 >moba建立user session_19、springcloud分散式Session之Spring Session

moba建立user session_19、springcloud分散式Session之Spring Session

技術標籤:moba建立user session

HttpSession是通過Servlet容器建立和管理的,像Tomcat/Jetty都是儲存在記憶體中的。但是把應用搭建成分散式的叢集,然後利用F5、LVS或Nginx做負載均衡,那麼來自同一使用者的Http請求將有可能被分發到多個不同的伺服器中。那問題來了,如何保證不同的伺服器能夠共享同一份session資料呢?最簡單的想法,就是把session資料儲存到記憶體以外的一個統一的地方,例如Memcached/Redis等資料庫中。那問題又來了,如何替換掉Servlet容器建立和管理的HttpSession的實現呢?

方案一:利用Servlet容器提供的外掛功能,自定義HttpSession的建立和管理策略,並通過配置的方式替換掉預設的策略。這方面其實早就有開源專案了,例如memcached-session-manager(可以參考負載均衡+session共享(memcached-session-manager實現),以及tomcat-redis-session-manager。不過這種方式有個缺點,就是需要耦合Tomcat/Jetty等Servlet容器的程式碼。

方案二:設計一個Filter,利用HttpServletRequestWrapper,實現自己的 getSession()方法,接管建立和管理Session資料的工作。spring-session就是通過這樣的思路實現的。

8b5af53e06c7079565480e8d838fe326.png

1、新建專案sc-redis-session,對應的pom.xml檔案如下

 4.0.0spring-cloud sc-redis-session 0.0.1-SNAPSHOTjarsc-redis-sessionhttp://maven.apache.orgorg.springframework.boot spring-boot-starter-parent 2.0.4.RELEASEorg.springframework.cloud spring-cloud-dependencies Finchley.RELEASEpomimportUTF-81.81.8org.springframework.boot spring-boot-starter-data-redis org.apache.commons commons-pool2 org.springframework.boot spring-boot-starter-web org.springframework.session spring-session-data-redis com.alibaba fastjson 1.2.49

2、新建springboot啟動類RedisSessionApplication.java

package com.redis.session;import org.springframework.boot.SpringApplication;import org.springframework.boot.autoconfigure.SpringBootApplication;import org.springframework.session.data.redis.config.annotation.web.http.EnableRedisHttpSession;@[email protected]
class RedisSessionApplication { public static void main(String[] args) { SpringApplication.run(RedisSessionApplication.class, args); }}

說明:主要使用EnableRedisHttpSession註解開啟spring分散式session,該類的作用是配置org.springframework.session.web.http.SessionRepositoryFilter進行請求攔截

3、新建配置檔案application.yml

 server: port: 7200spring: session: store-type: redis application: name: sc-redis-session redis: host: 127.0.0.1 password:  port: 6379 timeout: 10000 # 連線超時時間(毫秒) database: 0 # Redis預設情況下有16個分片,這裡配置具體使用的分片,預設是0 lettuce: pool: max-active: 8 # 連線池最大連線數(使用負值表示沒有限制) 預設 8 max-wait: -1 # 連線池最大阻塞等待時間(使用負值表示沒有限制) 預設 -1 max-idle: 8 # 連線池中的最大空閒連線 預設 8 min-idle: 0 # 連線池中的最小空閒連線 預設 0

4、 新建一個UserController

package com.redis.session.controller;import java.util.HashMap;import java.util.Map;import javax.servlet.http.Cookie;import javax.servlet.http.HttpServletRequest;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.ResponseBody;import org.springframework.web.bind.annotation.RestController;import com.alibaba.fastjson.JSON;import com.redis.session.model.User;@RestControllerpublic class UserController { private static final Logger logger = LoggerFactory.getLogger(UserController.class); /** * 把使用者放入session *  * @param request * @return */ @RequestMapping("/user/login") @ResponseBody public Map login(HttpServletRequest request) { // 取出session中的userName Object loginUser = request.getSession().getAttribute("loginUser"); if (loginUser == null) { logger.info("不存在session,設定user"); User u = new User(); u.setAge(23); u.setId(1L); u.setPosition("總裁"); u.setUserName("huangjinjin"); request.getSession().setAttribute("loginUser