Redis实现Session共享详解

Redis实现Session共享详解

Redis实现Session共享

这几天在做session共享这么一个小模块,也查了好多资料,给我的感觉,就是太乱了,一直找不到我想要的东西,几乎全部实现方法都与我的想法不一样,在这里,我总结一下自己是如何用Redis实现session共享的,方便自己以后查询,也希望能给有这方面需求的朋友一些帮助。

相关专题推荐:php session (包含图文、视频、案例)

先说一下我的开发环境:nginx、redis、tomcat,用moven构建项目,jetty服务器运行,所以在这里,下面也会涉及一下如何用maven打war包,部署在tomcat上运行。

redis是一个key-value数据库,存值取值,全靠这个key了,这里啰嗦一句,因为原创,专业的介绍我就不粘贴了,想了解的官方介绍的可以自行search.

 pom.xml中配置:

    redis.clients  jedis  2.8.1      org.springframework.data  spring-data-redis  1.7.2.RELEASE  

登录后复制

aplicationContext-redis.xml中配置

                                                                                                                                                                                                                                      

登录后复制

配置完毕后,开始代码实现:

在LoginController里:

第一步,引入RedisTemplate

@Autowired  @Qualifier("writeRedisTemplate")  private StringRedisTemplate writeTemplate;

登录后复制

这里只需要引入writeRedisTemplate即可,在登陆的时候,只负责写,只有在再次刷新的时候,经过过滤器,才需要读

第二步,正常登陆流程,登陆成功之后,request还要保存session信息

Redis实现Session共享详解

第三步,设置cookie值,把作为保存userSession信息在redis中的key值存入cookie,刷新浏览器的时候,过滤器可以从cookie中取到key值,进而去redis取对应的value值,即userSession

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

登录后复制

这里cookie跨域setDomain和setPath设置

第四步,把userSession信息存入redis中

RedisTemplate中写入redis的值要为String类型,需要把userSession对象转成Json字符串

userSessionString = JSON.toJSONString(userSession);

登录后复制

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

      net.sf.json-lib      json-lib      2.3      jdk15  

登录后复制

写入redis的代码如下:

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

登录后复制

完成这一操作,用户的session信息已经存入到redis中,可在redis中查看是否存入。

第五步:进入页面后,刷新页面,请求会经过过滤器,在Filter.Java中读取redis的值并进行一些处理

在过滤器这里,就无法通过注解的方式引入redisTemplate,可以通过如下的方式引入:

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

登录后复制

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

String cookid=MD5Util.MD5Encode("uasLoginer", "UTF-8");  Cookie[] cookies = req.getCookies();  String redisSessionKey = "";  if(cookies != null){      for (Cookie cookie : cookies) {          if(cookie.getName().equals(cookid)){              redisSessionKey = cookie.getValue() ;          }      }  }  UserSession userSession = null;  String userSessionString = (String) readTemplate.boundHashOps(UasContants.REDIS_USER_SESSION_KEY+"_"+redisSessionKey).get(redisSessionKey);  if(null != userSessionString ){      @SuppressWarnings("static-access")      JSONObject obj = new JSONObject().fromObject(userSessionString);//将json字符串转换为json对象      userSession = (UserSession)JSONObject.toBean(obj,UserSession.class);      writeTemplate.expire(UasContants.REDIS_USER_SESSION_KEY+"_"+redisSessionKey, 1800L, TimeUnit.SECONDS);      request.getSession().setAttribute(UasContants.USER_SESSION, userSession);  }  if (userSession != null) {      chain.doFilter(req, res);      return;  }else {      res.sendRedirect(UasContants.LOGIN_URL);      return;  }

登录后复制

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

        org.springframework.web.context.ContextLoaderListener          loginFilter      com.sfbest.uas.filter.LoginFilter                excludePaths          /login,/user/login,/user/auth                loginFilter      /*  

登录后复制

按照上面的配置,就可以用redis实现session共享的功能,但我在开发的时候,遇到一个蛋疼的问题,在测试环境上,

把项目部署在两台tomcat服务器上的时候,cookie里一直存不进去redis的key值,单台可以存进去,经过长期的检测,

终于发现是nginx配置出的问题,引以为戒,深深的阴影。下面我贴出我正常运行时nginx的配置代码

upstream uassessiontest.d.com {          server 10.103.16.226:8088;          server 10.103.16.226:8089;          }         server {                   log_format  sf_uastest  '$remote_addr - $remote_user [$time_local] "$request" '                                          '$status $body_bytes_sent "$http_referer" '                                          '"$http_user_agent" $http_cookie';                  listen 80;                  server_name uassessiontest.d.com;                  access_log /var/log/nginx/uassessiontest.log sf_uastest;                   location / {                          rewrite ^/$ /uas/ break;                          proxy_pass http://uassessiontest.d.com;                  }          }

登录后复制

红色的为当初少配的部分,这些部分是的作用是往浏览器端写入cookie值。

相关学习推荐:redis视频教程

以上就是Redis实现Session共享详解的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2042662.html

(0)
上一篇 2025年2月24日 00:45:11
下一篇 2025年2月18日 07:41:14

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • Redis是单线程的原因及高并发快的3大原因详解

    下面由redis教程栏目给大家介绍redis是单线程的原因及高并发快的3大原因详解,希望对需要的朋友有所帮助! Redis的高并发和快速原因 1.redis是基于内存的,内存的读写速度非常快;2.redis是单线程的,省去了很多上下文切换线…

    2025年2月24日
    200
  • Redis下载并安装

    下面由redis教程栏目给大家介绍redis下载、redis安装的具体方法步骤,希望对需要的朋友有所帮助! Redis下载并安装 Redis在GitHub上有一个活跃的社区。这些年来,大量的pull request被提出和合并,作者Anti…

    2025年2月24日
    200
  • redis中set和hset的区别和使用场景

    下面由redis教程栏目给大家介绍redis中set和hset的区别和使用场景,希望对需要的朋友有所帮助! redis中存数据时,到底什么时候用  hset 相比于 set 存数据时又有什么不一样? set  就是普通的已key-value…

    2025年2月24日
    200
  • 如何使用Redis实现排行榜功能

    下面由redis教程栏目给大家介绍使用redis实现排行榜功能的方法,希望对需要的朋友有所帮助! 排行榜功能是一个很普遍的需求。使用 Redis 中有序集合的特性来实现排行榜是又好又快的选择。 一般排行榜都是有实效性的,比如“用户积分榜”。…

    2025年2月24日
    200
  • php redis批量删除key的方法

    php redis删除key的方法:首先使用scan命令增量迭代的方式遍历出以【XX】为前缀的key;然后通过unlink函数删除指定的key即可。 推荐:《redis教程》 php redis批量删除key 摘要 使用scan命令增量迭代…

    2025年2月24日
    200
  • redis如何设置开机自启动

    下面由redis教程栏目给大家介绍使用redis实现排行榜功能的方法,希望对需要的朋友有所帮助! 1、拷贝 redis 安装目前下的 /usr/local/redis-4.0.8/utils/redis_init_script 到 /etc…

    2025年2月24日
    200
  • 详解Redis中5种数据结构的使用场景

    下面由redis教程栏目给大家介绍redis中5种数据结构的使用场景,本文对redis中的5种数据类型string、hash、list、set、sorted set做了讲解,需要的朋友可以参考下! 一、redis 数据结构使用场景 原来看过…

    2025年2月24日
    200
  • 2023年最新redis面试题大全及答案(收藏)

    因为Redis出色的高性能和并发,在大流量网站中必须要要用到的缓存技术,如果你想技术提升或者换更有挑战性的工作,那Redis是肯定需要掌握的! 今天【创想鸟】为同学们总结了一些redis面试中常被问到的问题,除了这些redis常见面试题,我…

    2025年2月24日
    200
  • PHP redis队列如何实现历史搜索功能

    下面由redis教程栏目给大家介绍redis队列实现历史搜索功能的方法,希望对需要的朋友有所帮助! redis队列实现历史搜索功能 1、使用phpinfo()函数查看你的配置,匹配对应的redis扩展版本(注:以下案例为window系统下进…

    2025年2月24日 数据库
    200
  • CentOS将Redis配置为系统服务的方法

    下面由redis教程栏目给大家介绍redis队列实现历史搜索功能的方法,希望对需要的朋友有所帮助! CentOS 将 Redis 配置为系统服务 创建 redis.service cd  /usr/lib/systemd/systemtou…

    2025年2月24日
    200

发表回复

登录后才能评论