pom.xml中配置:

    1. <!-- redis -->
    2. <dependency>
    3. <groupId>redis.clients</groupId>
    4. <artifactId>jedis</artifactId>
    5. <version>2.8.1</version>
    6. </dependency>
    7. <dependency>
    8. <groupId>org.springframework.data</groupId>
    9. <artifactId>spring-data-redis</artifactId>
    10. <version>1.7.2.RELEASE</version>
    11. </dependency>

    aplicationContext-redis.xml中配置

    1. <!-- redis 客户端配置 -->
    2. <bean id="jedisPoolConfig" class="redis.clients.jedis.JedisPoolConfig">
    3. <property name="maxTotal" value="${jedisPoolConfig.maxTotal}"/>
    4. <property name="maxIdle" value="${jedisPoolConfig.maxIdle}"/>
    5. <property name="maxWaitMillis" value="${jedisPoolConfig.maxWaitMillis}"/>
    6. <property name="testWhileIdle" value="true"/>
    7. <property name="testOnBorrow" value="false"/>
    8. <property name="testOnReturn" value="false"/>
    9. </bean>
    10. <bean id="readJedisConnectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory">
    11. <property name="hostName" value="${jedis.read.host}" />
    12. <property name="port" value="${jedis.read.port}" />
    13. <property name="password" value="${jedis.read.password}" />
    14. <property name="timeout" value="${jedis.read.timeout}" />
    15. <property name="database" value="${jedis.read.database}" />
    16. <property name="poolConfig" ref="jedisPoolConfig" />
    17. </bean>
    18. <bean id="writeJedisConnectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory">
    19. <property name="hostName" value="${jedis.write.host}" />
    20. <property name="port" value="${jedis.write.port}" />
    21. <property name="password" value="${jedis.write.password}" />
    22. <property name="timeout" value="${jedis.write.timeout}" />
    23. <property name="database" value="${jedis.write.database}" />
    24. <property name="poolConfig" ref="jedisPoolConfig" />
    25. </bean>
    26. <bean id="readRedisTemplate" class="org.springframework.data.redis.core.StringRedisTemplate">
    27. <property name="connectionFactory" ref="readJedisConnectionFactory" />
    28. </bean>
    29. <bean id="writeRedisTemplate" class="org.springframework.data.redis.core.StringRedisTemplate">
    30. <property name="connectionFactory" ref="writeJedisConnectionFactory" />
    31. </bean>

    配置完毕后,开始代码实现:
    在LoginController里:
    第一步,引入RedisTemplate

    1. @Autowired
    2. @Qualifier("writeRedisTemplate")
    3. private StringRedisTemplate writeTemplate;

    这里只需要引入writeRedisTemplate即可,在登陆的时候,只负责写,只有在再次刷新的时候,经过过滤器,才需要读
    第二步,正常登陆流程,登陆成功之后,request还要保存session信息
    redis实现session共享 - 图1
    第三步,设置cookie值,把作为保存userSession信息在redis中的key值存入cookie,刷新浏览器的时候,过滤器可以从cookie中取到key值,进而去redis取对应的value值,即userSession

    1. String domain = request.getServerName();
    2. String cookieId=MD5Util.MD5Encode("uasLoginer", "UTF-8");
    3. //生成token,用作session在redis存储中的key值
    4. StringredisSessionKey= UUID.randomUUID().toString();
    5. Cookie uasLoginer = new Cookie(cookieId, redisSessionKey);
    6. if (domain.startsWith("uas.")) {
    7. uasLoginer.setDomain(domain.substring(4,domain.length()));
    8. }else {
    9. uasLoginer.setDomain(domain);
    10. }
    11. uasLoginer.setMaxAge(60000);
    12. uasLoginer.setPath("/");
    13. response.addCookie(uasLoginer);

    这里cookie跨域setDomain和setPath设置
    第四步,把userSession信息存入redis中
    RedisTemplate中写入redis的值要为String类型,需要把userSession对象转成Json字符串

    1. userSessionString = JSON.toJSONString(userSession);

    在转Json的时候,遇到问题,导入import com.alibaba.fastjson.JSON;一直失败,发现pom中没有依赖Json的关系,如果有遇到相同的问题,可以检查下在pom.xml中是否有关于json的依赖关系,没的话,在pom.xml中导入json的依赖关系,如下:

    1. <dependency>
    2. <groupId>net.sf.json-lib</groupId>
    3. <artifactId>json-lib</artifactId>
    4. <version>2.3</version>
    5. <classifier>jdk15</classifier>
    6. </dependency>

    写入redis的代码如下:

    1. writeTemplate.opsForHash().put(UasContants.REDIS_USER_SESSION_KEY+"_"+redisSessionKey,redisSessionKey, userSessionString);
    2. writeTemplate.expire(UasContants.REDIS_USER_SESSION_KEY+"_"+redisSessionKey, 1800L, TimeUnit.SECONDS);//设置redis中值的有效期

    完成这一操作,用户的session信息已经存入到redis中,可在redis中查看是否存入。
    第五步:进入页面后,刷新页面,请求会经过过滤器,在Filter.java中读取redis的值并进行一些处理
    在过滤器这里,就无法通过注解的方式引入redisTemplate,可以通过如下的方式引入:

    1. BeanFactory beans = WebApplicationContextUtils.getWebApplicationContext(request.getSession().getServletContext());
    2. StringRedisTemplate readTemplate = (StringRedisTemplate) beans.getBean("readRedisTemplate");
    3. StringRedisTemplate writeTemplate = (StringRedisTemplate) beans.getBean("writeRedisTemplate");

    过滤器从cookie中取出redis的key值,用readTemplate读出value值

    1. String cookid=MD5Util.MD5Encode("uasLoginer", "UTF-8");
    2. Cookie[] cookies = req.getCookies();
    3. String redisSessionKey = "";
    4. if(cookies != null){
    5. for (Cookie cookie : cookies) {
    6. if(cookie.getName().equals(cookid)){
    7. redisSessionKey = cookie.getValue() ;
    8. }
    9. }
    10. }
    11. UserSession userSession = null;
    12. String userSessionString = (String) readTemplate.boundHashOps(UasContants.REDIS_USER_SESSION_KEY+"_"+redisSessionKey).get(redisSessionKey);
    13. if(null != userSessionString ){
    14. @SuppressWarnings("static-access")
    15. JSONObject obj = new JSONObject().fromObject(userSessionString);//将json字符串转换为json对象
    16. userSession = (UserSession)JSONObject.toBean(obj,UserSession.class);
    17. writeTemplate.expire(UasContants.REDIS_USER_SESSION_KEY+"_"+redisSessionKey, 1800L, TimeUnit.SECONDS);
    18. request.getSession().setAttribute(UasContants.USER_SESSION, userSession);
    19. }
    20. if (userSession != null) {
    21. chain.doFilter(req, res);
    22. return;
    23. }else {
    24. res.sendRedirect(UasContants.LOGIN_URL);
    25. return;
    26. }

    在这里,另外附上关于web.xml关于LoginFilter的配置,有需要的可以参考下:

    1. <!-- Spring监听器 -->
    2. <listener>
    3. <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    4. </listener>
    5. <filter>
    6. <filter-name>loginFilter</filter-name>
    7. <filter-class>com.sfbest.uas.filter.LoginFilter</filter-class>
    8. <init-param>
    9. <param-name>excludePaths</param-name>
    10. <param-value>/login,/user/login,/user/auth</param-value>
    11. </init-param>
    12. </filter>
    13. <filter-mapping>
    14. <filter-name>loginFilter</filter-name>
    15. <url-pattern>/*</url-pattern>
    16. </filter-mapping>

    按照上面的配置,就可以用redis实现session共享的功能,但我在开发的时候,遇到一个蛋疼的问题,在测试环境上,
    把项目部署在两台tomcat服务器上的时候,cookie里一直存不进去redis的key值,单台可以存进去,经过长期的检测,
    终于发现是nginx配置出的问题,引以为戒,深深的阴影。下面我贴出我正常运行时nginx的配置代码

    1. upstream uassessiontest.d.com {
    2. server 10.103.16.226:8088;
    3. server 10.103.16.226:8089;
    4. }
    5. server {
    6. log_format sf_uastest '$remote_addr - $remote_user [$time_local] "$request" '
    7. '$status $body_bytes_sent "$http_referer" '
    8. '"$http_user_agent" $http_cookie';
    9. listen 80;
    10. server_name uassessiontest.d.com;
    11. access_log /var/log/nginx/uassessiontest.log sf_uastest;
    12. location / {
    13. rewrite ^/$ /uas/ break;
    14. proxy_pass http://uassessiontest.d.com;
    15. }
    16. }