springboot session写进redis成功,但是另一台机器读取不到

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了springboot session写进redis成功,但是另一台机器读取不到相关的知识,希望对你有一定的参考价值。

我在一个服务器(41服务器)上部署了两个程序 session可以共享,分别是两个端口(8081,8084),但是部署在两台机器上(41服务器和42服务器)session就共享不了了, 谁知道为什么吗? 是不是要设置什么 redis是一台单独的服务器(43服务器)

在使用spring boot做负载均衡的时候,多个app之间的session要保持一致,这样负载到不同的app时候,在一个app登录之后,而打到另外一台服务器的时候,session丢失。
常规的解决方案都是使用:如apache使用mod_jk.conf。
在开发spring boot app的时候可以借助 spring session 和redis,用外置的redis来存储session的状态。
直接上代码,我这边直接默认你使用spring boot,如果你是普通的spring web项目,请参照 github.com/spring-projects/spring-session ,在spring boot配置更简单
1、增加repository到pom.xml
<repository>
<id>spring-milestone</id>
<url>repo.spring.io/libs-milestone</url>
</repository>

2、增加相关依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-redis</artifactId>
</dependency>

<dependency>
<groupId>org.springframework.session</groupId>
<artifactId>spring-session-data-redis</artifactId>
<version>1.0.0.RC1</version>
<type>pom</type>
</dependency>

3、RedisSessionConfig.java
package com.wisely.base;

import org.springframework.context.annotation.Configuration;
import org.springframework.session.data.redis.config.annotation.web.http.EnableRedisHttpSession;

@Configuration
@EnableRedisHttpSession
public class RedisSessionConfig



4、相关配置修改
在application.properties修改redis配置信息(请自行安装redis),请根据实际修改。如:
spring.redis.host=192.168.1.103

5、所有实体类实现Serializable接口
public class SysResource implements Serializable

6、查看效果
这时候登录系统在不同的app之间跳转的时候,session都是一致了,redis上可以看到:
7、总结
使用这些代码之后 ,无论你使用nginx或者apache,都无须在关心多个app之间的session一致的问题了。
参考技术A 我也是这个问题终于解决了,是要加上nginx负载均衡通过代理访问才能session共享,直接分别访问两台机器的地址是不能共享的。 参考技术B 两个浏览器代表两个不同的客户端,当然会有两个sessionId咯。
用户重复登录的问题可以通过注销上一个登录信息或者覆盖上一个登录的信息来解决。
由于http协议的无状态性,服务器是没法确定哪一个是你当前账号的session的...
-重新读了一遍问题,发现题主的意思应该是在服务器集群下如何使两个服务器辨别同一个用户才对,我觉得可以采用共同使用同一存储中心的方式来完成。恩,对于redis了解不多,希望有大神来指正,小的不才,说错勿怪。

SpringBoot初始教程之Redis集中式Session管理

SpringBoot初始教程之Redis集中式Session管理(四)

1.介绍

有关Session的管理方式这里就不再进行讨论,目前无非就是三种单机Session(基于单机内存,无法部署多台机器)、基于Cookie(安全性差)、基于全局的统一Session管理(redis、mysql)等多种方式
针对于像淘宝这种超大型网站来说Session如何管理的就无从得知了、但是可以通过yy的方式想象一下,这种大型架构都需要部署多台认证Server,但是一般来说集中式Session无法存储那么多的Session
那么就可以通过UID分片的形式来存储,不同UID分布在不同的Server上认证即可(纯属猜测丷)

2.快速开始

这里采用的是redis进行集中式Session管理,核心如下依赖

            <dependency>
                <groupId>org.springframework.session</groupId>
                <artifactId>spring-session</artifactId>
                <version>1.2.2.RELEASE</version>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-redis</artifactId>
            </dependency>

完整pom.xml


    <?xml version="1.0" encoding="UTF-8"?>
    <project xmlns="http://maven.apache.org/POM/4.0.0"
             xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
        <parent>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-parent</artifactId>
            <version>1.4.1.RELEASE</version>
        </parent>
        <modelVersion>4.0.0</modelVersion>

        <artifactId>springboot-4</artifactId>
        <dependencies>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-freemarker</artifactId>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter</artifactId>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-web</artifactId>
            </dependency>
            <dependency>
                <groupId>org.springframework.session</groupId>
                <artifactId>spring-session</artifactId>
                <version>1.2.2.RELEASE</version>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-redis</artifactId>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-test</artifactId>
                <scope>test</scope>
            </dependency>
        </dependencies>
        <build>
            <plugins>
                <plugin>
                    <groupId>org.springframework.boot</groupId>
                    <artifactId>spring-boot-maven-plugin</artifactId>
                    <version>1.4.1.RELEASE</version>
                </plugin>
            </plugins>
        </build>

    </project>

application.yaml配置

这块主要是通过application.yaml去配置redis的链接信息


    server:
      port: 8080
    spring:
      redis:
        database: 1
        host: localhost
        pool:
          max-active: 20

开启@EnableRedisHttpSession

通过加上@EnableRedisHttpSession注解,开启redis集中式session管理,所有的session都存放到了redis中


    @SpringBootApplication
    @EnableRedisHttpSession
    public class AppApplication 
        public static void main(String[] args) throws Exception 
            SpringApplication.run(AppApplication.class, args);
        
    

通过源码可知、可以通过设置maxInactiveIntervalInSeconds来设定session的统一过期时间,


    @Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
    @Target( java.lang.annotation.ElementType.TYPE )
    @Documented
    @Import(RedisHttpSessionConfiguration.class)
    @Configuration
    public @interface EnableRedisHttpSession 
        int maxInactiveIntervalInSeconds() default 1800;

        String redisNamespace() default "";


        RedisFlushMode redisFlushMode() default RedisFlushMode.ON_SAVE;
    

通过redis集中式管理session这种方式在使用上面对客户端是透明的,无需自己操作redis,在使用HttpSession对象的时候直接使用即可


    @RestController
    public class IndexController 
        @GetMapping("/index")
        public ResponseEntity index(HttpSession httpSession) 
            httpSession.setAttribute("user", "helloword");
            return ResponseEntity.ok("ok");
        

        @GetMapping("/helloword")
        public ResponseEntity hello(HttpSession httpSession) 
            return ResponseEntity.ok(httpSession.getAttribute("user"));
        
    

3. 其他扩展

SpringBoot Session集中式管理不仅仅限于redis这种方式,目前内部支持HttpSession with Pivotal GemFire、HttpSession with JDBC、HttpSession with Mongo
等多种方式具体可以参考http://docs.spring.io/spring-session/docs/1.2.2.RELEASE/reference/html5/#httpsession-jdbc
本文代码

https://git.oschina.net/wangkang_daydayup/SpringBoot-Learn/tree/master

springboot-4

以上是关于springboot session写进redis成功,但是另一台机器读取不到的主要内容,如果未能解决你的问题,请参考以下文章

springboot+redis实现session共享

springboot 整合 redis 共享Session-spring-session-data-redis

SpringSpringBoot + SpringSession + Redis 实现Session共享

SpringBoot初始教程之Redis集中式Session管理

SpringBoot初始教程之Redis集中式Session管理

springboot+spring session+redis+nginx实现session共享和负载均衡