<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          SpringBoot 一個(gè)騷操作搞定 session 共享,沒(méi)有比這更簡(jiǎn)單了

          共 3898字,需瀏覽 8分鐘

           ·

          2019-06-18 07:54

          轉(zhuǎn)自公眾號(hào):牧碼小子 作者:江南一點(diǎn)雨



          有的人可能會(huì)覺(jué)得題目有點(diǎn)夸張,其實(shí)不夸張,題目沒(méi)有使用任何修辭手法!認(rèn)真讀完本文,你就知道我說(shuō)的是對(duì)的了!


          在傳統(tǒng)的單服務(wù)架構(gòu)中,一般來(lái)說(shuō),只有一個(gè)服務(wù)器,那么不存在 Session 共享問(wèn)題,但是在分布式/集群項(xiàng)目中,Session 共享則是一個(gè)必須面對(duì)的問(wèn)題,先看一個(gè)簡(jiǎn)單的架構(gòu)圖:


          Java15 01.png


          在這樣的架構(gòu)中,會(huì)出現(xiàn)一些單服務(wù)中不存在的問(wèn)題,例如客戶端發(fā)起一個(gè)請(qǐng)求,這個(gè)請(qǐng)求到達(dá) Nginx 上之后,被 Nginx 轉(zhuǎn)發(fā)到 Tomcat A 上,然后在 Tomcat A 上往 session 中保存了一份數(shù)據(jù),下次又來(lái)一個(gè)請(qǐng)求,這個(gè)請(qǐng)求被轉(zhuǎn)發(fā)到 Tomcat B 上,此時(shí)再去 Session 中獲取數(shù)據(jù),發(fā)現(xiàn)沒(méi)有之前的數(shù)據(jù)。對(duì)于這一類問(wèn)題的解決,思路很簡(jiǎn)單,就是將各個(gè)服務(wù)之間需要共享的數(shù)據(jù),保存到一個(gè)公共的地方(主流方案就是 Redis):


          Java15 02.png


          當(dāng)所有 Tomcat 需要往 Session 中寫數(shù)據(jù)時(shí),都往 Redis 中寫,當(dāng)所有 Tomcat 需要讀數(shù)據(jù)時(shí),都從 Redis 中讀。這樣,不同的服務(wù)就可以使用相同的 Session 數(shù)據(jù)了。


          這樣的方案,可以由開發(fā)者手動(dòng)實(shí)現(xiàn),即手動(dòng)往 Redis 中存儲(chǔ)數(shù)據(jù),手動(dòng)從 Redis 中讀取數(shù)據(jù),相當(dāng)于使用一些 Redis 客戶端工具來(lái)實(shí)現(xiàn)這樣的功能,毫無(wú)疑問(wèn),手動(dòng)實(shí)現(xiàn)工作量還是蠻大的。


          一個(gè)簡(jiǎn)化的方案就是使用 Spring Session 來(lái)實(shí)現(xiàn)這一功能,Spring Session 就是使用 Spring 中的代理過(guò)濾器,將所有的 Session 操作攔截下來(lái),自動(dòng)的將數(shù)據(jù) 同步到 Redis 中,或者自動(dòng)的從 Redis 中讀取數(shù)據(jù)。


          對(duì)于開發(fā)者來(lái)說(shuō),所有關(guān)于 Session 同步的操作都是透明的,開發(fā)者使用 Spring Session,一旦配置完成后,具體的用法就像使用一個(gè)普通的 Session 一樣。


          1 實(shí)戰(zhàn)

          1.1 創(chuàng)建工程


          首先 創(chuàng)建一個(gè) Spring Boot 工程,引入 Web、Spring Session 以及 Redis:


          Java15 03.png


          創(chuàng)建成功之后,pom.xml 文件如下:

          1. org.springframework.boot

          2. spring-boot-starter-data-redis

          3. org.springframework.boot

          4. spring-boot-starter-web

          5. org.springframework.session

          6. spring-session-data-redis


          注意:

          這里我使用的 Spring Boot 版本是 2.1.4 ,如果使用當(dāng)前最新版 Spring Boot2.1.5 的話,除了上面這些依賴之外,需要額外添加 Spring Security 依賴(其他操作不受影響,僅僅只是多了一個(gè)依賴,當(dāng)然也多了 Spring Security 的一些默認(rèn)認(rèn)證流程)。


          1.2 配置 Redis


          1. spring.redis.host=192.168.66.128

          2. spring.redis.port=6379

          3. spring.redis.password=123

          4. spring.redis.database=0


          這里的 Redis ,我雖然配置了四行,但是考慮到端口默認(rèn)就是 6379 ,database 默認(rèn)就是 0,所以真正要配置的,其實(shí)就是兩行。


          1.3 使用


          配置完成后 ,就可以使用 Spring Session 了,其實(shí)就是使用普通的 HttpSession ,其他的 Session 同步到 Redis 等操作,框架已經(jīng)自動(dòng)幫你完成了:

          1. @RestController

          2. public class HelloController {

          3. @Value("${server.port}")

          4. Integer port;

          5. @GetMapping("/set")

          6. public String set(HttpSession session) {

          7. session.setAttribute("user", "javaboy");

          8. return String.valueOf(port);

          9. }

          10. @GetMapping("/get")

          11. public String get(HttpSession session) {

          12. return session.getAttribute("user") + ":" + port;

          13. }

          14. }


          考慮到一會(huì) Spring Boot 將以集群的方式啟動(dòng) ,為了獲取每一個(gè)請(qǐng)求到底是哪一個(gè) Spring Boot 提供的服務(wù),需要在每次請(qǐng)求時(shí)返回當(dāng)前服務(wù)的端口號(hào),因此這里我注入了 server.port 。


          接下來(lái) ,項(xiàng)目打包:


          Java15 04.png


          打包之后,啟動(dòng)項(xiàng)目的兩個(gè)實(shí)例:

          1. java -jar sessionshare-0.0.1-SNAPSHOT.jar --server.port=8080

          2. java -jar sessionshare-0.0.1-SNAPSHOT.jar --server.port=8081


          然后先訪問(wèn) localhost:8080/set 向 8080 這個(gè)服務(wù)的 Session 中保存一個(gè)變量,訪問(wèn)完成后,數(shù)據(jù)就已經(jīng)自動(dòng)同步到 Redis 中 了 :


          Java15 05.png


          然后,再調(diào)用 localhost:8081/get 接口,就可以獲取到 8080 服務(wù)的 session中的數(shù)據(jù):


          Java15 06.png


          此時(shí)關(guān)于 session 共享的配置就已經(jīng)全部完成了,session 共享的效果我們已經(jīng)看到了,但是每次訪問(wèn)都是我自己手動(dòng)切換服務(wù)實(shí)例,因此,接下來(lái)我們來(lái)引入 Nginx ,實(shí)現(xiàn)服務(wù)實(shí)例自動(dòng)切換。


          1.4 引入 Nginx


          很簡(jiǎn)單,進(jìn)入 Nginx 的安裝目錄的 conf 目錄下(默認(rèn)是在 /usr/local/nginx/conf),編輯 nginx.conf 文件:


          Java15 07.png


          在這段配置中:

          1. upstream 表示配置上游服務(wù)器

          2. javaboy.org 表示服務(wù)器集群的名字,這個(gè)可以隨意取名字

          3. upstream 里邊配置的是一個(gè)個(gè)的單獨(dú)服務(wù)

          4. weight 表示服務(wù)的權(quán)重,意味者將有多少比例的請(qǐng)求從 Nginx 上轉(zhuǎn)發(fā)到該服務(wù)上

          5. location 中的 proxy_pass 表示請(qǐng)求轉(zhuǎn)發(fā)的地址,?/?表示攔截到所有的請(qǐng)求,轉(zhuǎn)發(fā)轉(zhuǎn)發(fā)到剛剛配置好的服務(wù)集群中

          6. proxy_redirect 表示設(shè)置當(dāng)發(fā)生重定向請(qǐng)求時(shí),nginx 自動(dòng)修正響應(yīng)頭數(shù)據(jù)(默認(rèn)是 Tomcat 返回重定向,此時(shí)重定向的地址是 Tomcat 的地址,我們需要將之修改使之成為 Nginx 的地址)。


          配置完成后,將本地的 Spring Boot 打包好的 jar 上傳到 Linux ,然后在 Linux 上分別啟動(dòng)兩個(gè) Spring Boot 實(shí)例:

          1. nohup java -jar sessionshare-0.0.1-SNAPSHOT.jar --server.port=8080 &

          2. nohup java -jar sessionshare-0.0.1-SNAPSHOT.jar --server.port=8081 &


          其中

          • nohup 表示當(dāng)終端關(guān)閉時(shí),Spring Boot 不要停止運(yùn)行

          • & 表示讓 Spring Boot 在后臺(tái)啟動(dòng)


          配置完成后,重啟 Nginx:

          1. /usr/local/nginx/sbin/nginx -s reload


          Nginx 啟動(dòng)成功后,我們首先手動(dòng)清除 Redis 上的數(shù)據(jù),然后訪問(wèn) 192.168.66.128/set 表示向 session中保存數(shù)據(jù),這個(gè)請(qǐng)求首先會(huì)到達(dá) Nginx上,再由 Nginx 轉(zhuǎn)發(fā)給某一個(gè) SpringBoot 實(shí)例:


          Java15 08.png


          如上,表示端口為 8081 的 SpringBoot 處理了這個(gè) /set 請(qǐng)求,再訪問(wèn) /get 請(qǐng)求:


          Java15 09.png


          可以看到, /get 請(qǐng)求是被端口為 8080 的服務(wù)所處理的。


          2 總結(jié)


          本文主要向大家介紹了 Spring Session 的使用,另外也涉及到一些 Nginx 的使用 ,雖然本文較長(zhǎng),但是實(shí)際上 Spring Session 的配置沒(méi)啥。


          我們寫了一些代碼,也做了一些配置,但是全都和 Spring Session 無(wú)關(guān),配置是配置 Redis,代碼就是普通的 HttpSession,和 Spring Session 沒(méi)有任何關(guān)系!


          唯一和 Spring Session 相關(guān)的,可能就是我在一開始引入了 Spring Session 的依賴吧!


          如果大家沒(méi)有在 SSM 架構(gòu)中用過(guò) Spring Session ,可能不太好理解我們?cè)?Spring Boot 中使用 Spring Session 有多么方便,因?yàn)樵?SSM 架構(gòu)中,Spring Session 的使用要配置三個(gè)地方 ,一個(gè)是 web.xml 配置代理過(guò)濾器,然后在 Spring 容器中配置 Redis,最后再配置 Spring Session,步驟還是有些繁瑣的,而 Spring Boot 中直接幫我們省去了這些繁瑣的步驟!不用再去配置 Spring Session。


          好了 ,本文就說(shuō)到這里,有問(wèn)題歡迎留言討論 ,本文相關(guān)案例我已經(jīng)上傳到 GitHub ,大家可以自行下載:https://github.com/lenve/javaboy-code-samples




          瀏覽 59
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評(píng)論
          圖片
          表情
          推薦
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  男女操B| 欧美黄色宗合一级二级三级 | 草青青在线免费视频 | 久无码久无码AV无码 | 国产大学生一级A片 |