首页技术文章正文

基于SpringBoot+Redis的Session共享与单点登录【黑马web前端】

更新时间:2019年07月26日 10时54分44秒 来源:黑马程序员论坛

使用Redis来实现Session共享,其实网上已经有很多例子了,这是确保在集群部署中最典型的redis使用场景。在SpringBoot项目中,其实可以一行运行代码都不用写,只需要简单添加添加依赖和一行注解就可以实现(当然配置信息还是需要的)。
然后简单地把该项目部署到不同的tomcat下,比如不同的端口(A、B),但项目访问路径是相同的。此时在A中使用set方法,然后在B中使用get方法,就可以发现B中可以获取A中设置的内容。
但如果就把这样的一个项目在多个tomcat中的部署说实现了单点登录,那就不对了。
所谓单点登录是指在不同的项目中,只需要任何一个项目登录了,其他项目不需要登录。
同样是上面的例子,我们把set和get两个方法分别放到两个项目(set、get)中,并且以集群方式把两个项目都部署到服务器A和B中,然后分别访问A服务器的set和B服务器的get,你就会发现完全得不到你想要的结果。
同一项目中的set/get
依赖添加就不说了,直接使用最简单的方式
@SpringBootApplication@EnableRedisHttpSession@RestControllerpublic 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;    }}
将该项目打war包,分别部署在tomcatA(端口8080),tomcatB(端口8081),然后通过tomcatA/set 方法设置session,再使用 tomcatB/get 方法即可获得session的值。但这只是实现了同一项目session的共享。并不是单点登录。
为了验证,我们不仿将set/get方法拆分为两个项目。
拆分set/get为两个项目
  • get项目


@SpringBootApplication@EnableRedisHttpSession@RestControllerpublic 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;    }}
将该项目打包为set.war
  • set项目


@SpringBootApplication@EnableRedisHttpSession@RestControllerpublic 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;    }}
将该项目打包为get.war
再分别将set.war,get.war部署在tomcatA和tomcatB,再通过 tomcatA/set 设置session内容, 然后通过 tomcatB/get 就发现无法获得session的值。
问题分析
尽管我们使用的路径都是一样的,但其实是两个项目,与前面的一个项目是完全不同的,问题就在于 session和cookie在默认情况下是与项目路径相关的,在同一个项目的情况下两个方法所需要的cookie依赖的项目路径是相同的,所以获取session的值就没有问题,但在后一种情况下,cookie的路径是分别属于不同的项目的,所以第二个项目就无法获得第一个项目中设置的session内容了。
解决方法
解决方法在springboot项目中其实也非常简单。既然cookie路径发生了变化,那我们让它配置为相同的路径就解决了。
在每个子项目中都添加一个配置类或者直接设置cookie的路径,如果有域名还可以设置域名的限制,比如 set.xxx.com 与 get.xxx.com 这种情况与我们就需要设置cookie的域名为 xxx.com,以确保无法在哪个项目下都能够获取 xxx.com 这个域名下的cookie值。这样就确保能够正常获得共享的session值了。
@Configurationpublic class CookieConfig {    @Bean    public static DefaultCookieSerializer defaultCookieSerializer() {        DefaultCookieSerializer serializer = new DefaultCookieSerializer();        serializer.setCookiePath("/");        //serializer.setDomainName("xxx.com"); //如果使用域名访问,建议对这一句进行设置            return serializer;    }}
以上才是正直的redis实现单点登录的正确打开方式。

推荐了解热门学科

java培训 Python人工智能 Web前端培训 PHP培训
区块链培训 影视制作培训 C++培训 产品经理培训
UI设计培训 新媒体培训 产品经理培训 Linux运维
大数据培训 智能机器人软件开发




传智播客是一家致力于培养高素质软件开发人才的科技公司“黑马程序员”是传智播客旗下高端IT教育品牌。自“黑马程序员”成立以来,教学研发团队一直致力于打造精品课程资源,不断在产、学、研3个层面创新自己的执教理念与教学方针,并集中“黑马程序员”的优势力量,针对性地出版了计算机系列教材50多册,制作教学视频数+套,发表各类技术文章数百篇。

传智播客从未停止思考

传智播客副总裁毕向东在2019IT培训行业变革大会提到,“传智播客意识到企业的用人需求已经从初级程序员升级到中高级程序员,具备多领域、多行业项目经验的人才成为企业用人的首选。”

中级程序员和初级程序员的差别在哪里?
项目经验。毕向东表示,“中级程序员和初级程序员最大的差别在于中级程序员比初级程序员多了三四年的工作经验,从而多出了更多的项目经验。“为此,传智播客研究院引进曾在知名IT企业如阿里、IBM就职的高级技术专家,集中研发面向中高级程序员的课程,用以满足企业用人需求,尽快补全IT行业所需的人才缺口。

何为中高级程序员课程?

传智播客进行了定义。中高级程序员课程,是在当前主流的初级程序员课程的基础上,增加多领域多行业的含金量项目,从技术的广度和深度上进行拓展“我们希望用5年的时间,打造上百个高含金量的项目,覆盖主流的32个行业。”传智播客课程研发总监于洋表示。




黑马程序员热门视频教程【点击播放】

Python入门教程完整版(懂中文就能学会) 零起点打开Java世界的大门
C++| 匠心之作 从0到1入门学编程 PHP|零基础入门开发者编程核心技术
Web前端入门教程_Web前端html+css+JavaScript 软件测试入门到精通


在线咨询 我要报名
和我们在线交谈!