`
jaychang
  • 浏览: 736068 次
  • 性别: Icon_minigender_1
  • 来自: 嘉兴
社区版块
存档分类
最新评论

基于MSM实现nginx负载均衡+tomcat集群+memcached(session共享)

 
阅读更多
摘要: 基于 MSM(memcached-session-manager)做session共享来实现tomcat集群,并使用 nginx 实现负载均衡,负载均衡方案使用 nginx-sticky-module

Session共享的方案介绍

Session 共享有多种解决方法,常用的有四种:客户端 Cookie 保存、服务器间 Session 同步、使用集群管理 Session(如本文要介绍的Memcached Session Manager) 、把 Session 持久化到数据库:

  1. 客户端 Cookie 保存 以cookie加密的方式保存在客户端.优点是减轻服务器端的压力,每次session信息被写在客服端,然后经浏览器再次提交到服务器。即使两次请求在集群中的两台服务器上完成,也可以到达session共享。
  2. 将 session 持久化到数据中 这种共享session的方式即将session信息存入数据库中,其它应用可以从数据库中查出 session 信息。目前采用这种方案时所使用的数据库一般为mysql。 利用数据库共享 session 的方案有一定的实用性,但也有如下缺点:首先 session 的并发读写在数据库中完成,对 mysql 的性能要求比较高;其次,我们需要额外地实现 session 淘汰(超时)逻辑代码,即定时从数据库表中更新和删除 session 信息,增加了工作量。
  3. 使用服务器间 session 同步 使用主-从服务器的架构,当用户在主服务器上登录后,通过脚本或者守护进程的方式,将 session 信息传递到各个从服务器中,这样用户访问其它的从服务器时,就可以读到session信息。 缺点:比如速度慢、不稳定等,另外,如果 session 信息传递是主->从单向的,会有一些风险,比如主服务器down了,其它服务器无法获得 session 信息
  4. 使用集群统一管理Session 提供一个集群保存 session 共享信息.其他应用统统把自己的 session 信息存放到 session 集群服务器组。当应用系统需要 session 信息的时候直接到 session 集群服务器上读取。目前大多都是使用 Memcache 来对 Session 进行存储。 以 Memcache 来实现 Session 共享的方式目前比较流行的有两种实现方案,下面主要对这两种方案进行介绍。

    使用Filter方式: 此方式使用过滤器的方式重新对httpRequest 对象进行了包装,并加入memcached客户端,此方式的优点是:使用简单,把过滤器配置进去即可,另外比较灵活,因为它是在客户端实现的,配置比较灵活,而且服务器无关,你可以在任何支持servlet的容器上部署。

    使用memcached-session-manager方式: memcached-session-manager,俗称 MSM ,是一个用于解决分布式 tomcat 环境下 session 共享的问题的开源解决方案。它的实现原理为以tomcat插件的方式部署在服务器,修改了 servlet 容器代码中的 session 相关代码,使其连接 memcached ,在 memcached 中创建和更新session。MSM拥有如下特性:

    支持Tomcat6、Tomcat7、Tomcat8
    支持黏性、非黏性 Session
    无单一故障点
    可处理 tomcat 故障转移
    可处理 memcached 故障转移
    插件式 session 序列化
    允许异步保存 session ,以提升响应速度
    只有当 session 有修改时,才会将 session 写回 memcached
    JMX 管理&监控
    

    Memcached-session-manager 支持tomcat6、tomcat7、tomcat8 ,利用 Value(Tomcat 阀)对 Request 进行跟踪。 Request 请求到来时,从 memcached 加载 session , Request 请求结束时,将 tomcat session 更新至 memcached ,以达到 session 共享之目的, 支持 sticky 和 non-sticky 模式。

    优点:开发者不用考虑session共享的问题了,可以专注于业务逻辑开发,像正常使用 session 那样使用就完事了。不用显示编写代码,只需要对服务器进行配置即可使用。

    缺点:如果你想改变session策略的话,必须重新部署每个服务器的servlet容器。

    为什么会有MSM(memcached-session-manager)

    我们都知道对于一些比较大型的网站,在正式部署时一般是部署在不同故障域的多台应用服务器上,以 JavaEE 应用为例,一般我们都会部署在 tomcat 下,假如我们部署了10台 tomcat 服务器,那这10台 tomcat 可能是部署在不同的机器上,然后将应用程序copy到这10台 tomcat 下,然后启动所有 tomcat ,一般来说这样做的目的是为了达到负载均衡以及避免单点故障,另外也考虑到国内网络环境的原因,避免跨网络运营商访问而导致访问速度低下的问题,当然不要忘了坐镇这10台 tomcat 前面的还有我们的反向代理服务器,比如 nginx ,我今天主要讲的是,对于这种分布式 tomcat 环境,我们如何保证 session 的唯一性(也可以说是 session 的共享)。这也是在目前的很多项目中需要解决的一个问题,当然实际上这并不是什么新的议题,之前就有很多解决方案,但是一般来说的大体的解决方案是自己通过编写一段代码或者通过配置 tomcat 的 filter ,将产生的 session 放到同一个内存数据库中,事实上这确实可行的,只不过我比较懒,我总是觉得这种问题应该有更省事更成熟的解决方案,那确实是有的,也就是我马上介绍的 Memcached Session Manager,简称 msm ,这就是一个用于解决分布式 tomcat 环境下 session 共享的问题的开源解决方案。

memcached-session-manager解决的问题

想象下web应用程序运行在多个tomcat,期望session能实现故障转移。你需要一个可扩展的方案-仅仅增加tomcat来处理更多的会话。该方案可通过memcached节点存储备份会话来实现。当一个tomcat负载过重或挂掉时,其他tomcat就会接管这个tomcat,从相应的memcached节点中获取会话数据,之后就可以服务这个会话。当然多个tomcat前面还需要有一个负载均衡,比如nginx。

memcached-session-manager是如何工作的

首先谈下tomcat故障转移

msm安装在tomcat里,tomcat会在本地保留所有会话信息就像StandardManager一样。 此外,一个请求完成后,session会被备份到memcached节点。 当服务同一会话的下一次请求时,tomcat可以在本地找到这个会话数据,同一会话的第二次请求 处理完后,会话数据会更新到memcached节点。 假设处理某个会话的tomcat挂了。 那么下次请求会被路由到另一个tomcat。而这个tomcat没有在本地保存该会话的数据。因此它 会去相应的memcached(根据请求头中sessionid的后缀,后面配置$CATALINA_HOME/conf/context.xml时,memcachedNodes="n1:localhost:11211,n2:localhost:11212",就是n1,n2)中查找此次请求的会话数据并保存到本地。 这样这个tomcat就可以处理此次会话了。当这个tomcat处理完此次会话,它会将更新相应memcached节点存储的session信息。

输入图片说明

注:上图8 tomcat1故障,路由到tomcat2由负载均衡完成(如nginx)。

再谈下memcahced故障转移

msm也实现了memcached的故障转移。当一个memcached节点不可用时,session信息就会被转移到其他memcached节点。 与此同时,sessionid会被修改,一个新的JESESSIONID(响应头会有Set-Cookie:JESSIONID;XXXXXXXXXXXXX)会被发送 到浏览器端。当你使用sticky session时,确保你的负载均衡不会给sessionid添加后缀。

MSM sticky模式工作原理

MSM(memcached-session-manager) 支持tomcat6 和tomcat7 ,利用 Value(Tomcat 阀)对Request进行跟踪。Request请求到来时,从memcached加载session,Request请求结束时,将tomcat session更新至memcached,以达到session共享之目的, 支持 sticky 和 non-sticky 模式。需要注意的是使用sticky模式时需要配置jvmroute参数,配置方式如下:

配置$CATALINA_HOME/conf/server.xml

<Engine name="Catalina"defaultHost="localhost"jvmRoute="tomcatx">

注意每台tomcat的jvmroute参数都不能一样

Sticky 模式:tomcat本地session 为 主session, memcached 为备 session。Request请求到来时, 从memcached加载备 session 到 tomcat (仅当tomcat jvmroute发生变化时,否则直接取tomcat本地session);Request请求结束时,将tomcat本地session更新至memcached,以达到主备同步之目的。下面是sticky模式时响应的流程图(图片来源网络):

输入图片说明

non-sticky模式本文不介绍。详见:http://gong1208.iteye.com/blog/1596120

集群的拓扑图

输入图片说明

为何在两个tomcat前面要放一个nginx

  1. 首先nginx可以作为两个tomcat的负载均衡,均衡两个tomat负载压力,负载均衡也可以使得客户端访问可以使用统一的url,如果没有nginx那么,访问tomcat1必须要用http://domain:8001/,而访问tomcat2需要用http://domain:8002/
  2. nginx处理静态资源的性能比tomcat好很多

由于nginx需要安装sticky session,故需要在linux环境下完成

准备工作

  1. 下载tomcat,nginx,memcached,msm(http://repo1.maven.org/maven2/de/javakaffee/msm/),需要根据tomcat的版本下载相应版本的msm,由于用到了memcached,因此还要用到memcached的java api包,```http://mvnrepository.com/artifact/de.javakaffee.msm/memcached-session-manager
    http://mvnrepository.com/artifact/de.javakaffee/kryo-serializers和kryo
    
    http://repo1.maven.org/maven2/com/googlecode/kryo/1.04/,还需要下载minlog```http://mvnrepository.com/artifact/com.googlecode/minlog/1.2
    ```,asm
    
    http://mvnrepository.com/artifact/asm/asm
    http://mvnrepository.com/artifact/com.googlecode/reflectasm/1.01
    
    ```
    2015/12/07  15:09            15,979 annotations-api.jar
    2016/01/11  15:51            43,581 asm-3.3.1.jar
    2015/12/07  15:09            55,011 catalina-ant.jar
    2015/12/07  15:09           131,075 catalina-ha.jar
    2015/12/07  15:09           260,900 catalina-tribes.jar
    2015/12/07  15:09         1,651,858 catalina.jar
    2015/12/07  15:09         2,310,271 ecj-4.4.2.jar
    2015/12/07  15:09            55,505 el-api.jar
    2015/12/07  15:09           124,695 jasper-el.jar
    2015/12/07  15:09           601,087 jasper.jar
    2015/12/07  15:09            87,805 jsp-api.jar
    2016/01/12  09:38            94,830 kryo-1.04.jar
    2016/01/12  09:30            62,112 kryo-serializers-0.11.jar
    2016/01/11  16:50           147,025 memcached-session-manager-1.8.3.jar
    2016/01/11  15:44            11,284 memcached-session-manager-tc7-1.8.3.jar
    2016/01/12  09:39             4,879 minlog-1.2.jar
    2016/01/11  15:42            29,328 msm-kryo-serializer-1.8.3.jar
    2016/01/12  09:42            11,615 reflectasm-1.01.jar
    2015/12/07  15:09           198,017 servlet-api.jar
    2016/01/11  16:31           467,218 spymemcached-2.11.7.jar
    2015/12/07  15:09             6,522 tomcat-api.jar
    2015/12/07  15:09           790,612 tomcat-coyote.jar
    2015/12/07  15:09           234,043 tomcat-dbcp.jar
    2015/12/07  15:09            71,860 tomcat-i18n-es.jar
    2015/12/07  15:09            43,793 tomcat-i18n-fr.jar
    2015/12/07  15:09            47,036 tomcat-i18n-ja.jar
    2015/12/07  15:09           127,483 tomcat-jdbc.jar
    2015/12/07  15:09            32,893 tomcat-util.jar
    2015/12/07  15:09           214,782 tomcat7-websocket.jar
    2015/12/07  15:09            36,271 websocket-api.jar
    

配置tomcat

修改两个tomcat的$CATALINA_HOME/conf/server.xml

  1. tomcat1
    <Servicename="Catalina">
     <Connectorport="8001"protocol="HTTP/1.1"connectionTimeout="20000"redirectPort="8443" />
     <Connectorport="8009"protocol="AJP/1.3"redirectPort="8443" />
     <Enginename="Catalina"defaultHost="localhost"jvmRoute="tomcat1">
       <RealmclassName="org.apache.catalina.realm.LockOutRealm">
         <RealmclassName="org.apache.catalina.realm.UserDatabaseRealm"resourceName="UserDatabase"/>
       </Realm>
       <Hostname="localhost"unpackWARs="true"autoDeploy="true">
         <ValveclassName="org.apache.catalina.valves.AccessLogValve"directory="logs"prefix="localhost_access_log."suffix=".txt"pattern="%h %l %u %t &quot;%r&quot; %s %b" />
         <ContextdocBase="E:/HPCWorkFolder/ServerCluster/nginx-tomcat-memcached-manager-session/www/webapptomcat1"path=""reloadable="true" />
       </Host>
     </Engine>
    </Service>
    

tomcat1 Engine标签的jvmRoute属性值配置为tomcat1

  1. tomcat2
  <Servicename="Catalina">
    <Connectorport="8002"protocol="HTTP/1.1"connectionTimeout="20000"redirectPort="8443" />
    <Connectorport="8019"protocol="AJP/1.3"redirectPort="8443" />
    <Enginename="Catalina"defaultHost="localhost"jvmRoute="tomcat2">
      <RealmclassName="org.apache.catalina.realm.LockOutRealm">
        <RealmclassName="org.apache.catalina.realm.UserDatabaseRealm"resourceName="UserDatabase"/>
      </Realm>
      <Hostname="localhost"unpackWARs="true"autoDeploy="true">
        <ValveclassName="org.apache.catalina.valves.AccessLogValve"directory="logs"prefix="localhost_access_log."suffix=".txt"pattern="%h %l %u %t &quot;%r&quot; %s %b" />
        <ContextdocBase="E:/HPCWorkFolder/ServerCluster/nginx-tomcat-memcached-manager-session/www/webapptomcat2"path=""reloadable="true" />
      </Host>
    </Engine>
  </Service>

tomcat2 Engine标签的jvmRoute属性值配置为tomcat2

修改$CATALINA_HOME/conf/context.xml

tomcat1:

  <ManagerclassName="de.javakaffee.web.msm.MemcachedBackupSessionManager"memcachedNodes="n1:localhost:11211,n2:localhost:11212"failoverNodes="n1"requestUriIgnorePattern=".*\.(ico|png|gif|jpg|css|js)$"transcoderFactoryClass="de.javakaffee.web.msm.serializer.kryo.KryoTranscoderFactory"
    />

tomcat2: 另一个failoverNodes="n2"

<ManagerclassName="de.javakaffee.web.msm.MemcachedBackupSessionManager"memcachedNodes="n1:localhost:11211,n2:localhost:11212"failoverNodes="n2"requestUriIgnorePattern=".*\.(ico|png|gif|jpg|css|js)$"transcoderFactoryClass="de.javakaffee.web.msm.serializer.kryo.KryoTranscoderFactory"/>

意思是tomcat1优先将session存到memcached1,tomcat2优先将session存到memchaced2

开启两个memcached,tomcat

memcached -p11211 -m32 memcached -p11212 -m32

开启两个tomcat

tomcat1终端显示:

-  finished initialization:-sticky:true- operation timeout:1000- node ids: [n2]
- failover node ids: [n1]
- storage key prefix: null
--------

tomcat2终端显示:

-  finished initialization:-sticky:true- operation timeout:1000- node ids: [n1]
- failover node ids: [n2]
- storage key prefix: null
--------
--------

表示tomcat1,tomcat2,配置成功了

nginx安装配置见

http://my.oschina.net/u/1167421/blog/604633

验证(memcached-session-manager是如何工作的)

在tomcat1的web应用目录下放一个index.jsp文件,内容如下: tomcat1<br> <%=request.getSession(true).getId()%>

在tomcat2的web应用目录下放一个index.jsp文件,内容如下: tomcat2<br> <%=request.getSession(true).getId()%>

第1次请求 输入图片说明

看到响应头中SetCookie

JSESSIONID=786579866D7914523416D9C35A3F74DB-n2.tomcat1 以及route(由nginx设置的)

第2次请求 输入图片说明

看到响应头中已经没有Set-Cookie

第3次请求(在浏览器端发送第3次请求前,关掉tomcat1) 输入图片说明

JSESSIONID=786579866D7914523416D9C35A3F74DB-n1.tomcat2 以及route,其中route值已经变了跟第1次请求相比

以上结果说明,tomcat1优先将session存储到n2,tomcat2优先将session存储到n1,第2次请求,tomcat1挂掉,请求转交tomcat2处理,tomcat2根据SESSIONID后缀n2去memcached2查找SESSSION信息,修改JESESSIONID的后缀,由786579866D7914523416D9C35A3F74DB-n2.tomcat1变成786579866D7914523416D9C35A3F74DB-n1.tomcat2。前缀还是786579866D7914523416D9C35A3F74DB。做到了SESSION保持。

参考资料

  1. memcached-session-manager源码及官方文档
  2. memcached-session-manager配置
  3. 使用memcached实现的tomcat高可用集群
  4. msm的jar包下载
  5. nginx服务器安装及配置文件详解
  6. 使用nginx sticky模块实现基于cookie的负载均衡
  7. 集群中几种session同步解决方案的比较
  8. MSM--Memcached_Session_Manager介绍及使用

 

分享到:
评论

相关推荐

    Nginx+Tomcat7+Memcached集群Session共享

    Nginx+Tomcat7+Memcached集群Session共享 完整例子 主要是利用memcached-session-manager(下面简称msm)开源tomcat插件改变tomcat原始的session存储机制,将session的存储放到分布式缓存memcache中,从而实现对...

    nginx+tomcat7+memcached session会话保持

    本文将详细介绍如何在Nginx、Tomcat7和Memcached的环境下实现会话保持,确保用户在多台服务器之间切换时仍能保持其会话状态。 首先,我们要理解会话保持的重要性。在分布式系统中,用户可能与集群中的任何一台...

    windows下Nginx+memcached+Tomcat集群配置实现session共享与负载均衡

    【Nginx、memcached和Tomcat集群配置实现session共享与负载均衡】 在Windows环境下,构建一个基于Nginx的高可用性Web集群,通常包括Nginx作为反向代理和负载均衡器,memcached用于session共享,以及多个Tomcat实例...

    Windows7 + Nginx + Memcached + Tomcat 集群 session 共享

    本主题将详细探讨如何在Windows7环境下,利用Nginx作为负载均衡器,Memcached作为session存储,以及Tomcat作为应用服务器,实现session共享。 **1. Nginx介绍与配置** Nginx是一款高性能的HTTP和反向代理服务器,它...

    nginx+tomcat集群+memcached实现session共享(适用tc7、tc8)

    兼容Tomcat7 与tomcat8 +memcached做session共享 , 解决了tomcat7与Tomcat8的 java.lang.NoSuchFieldError: attributes 错误。

    Nginx+Tomcat+Memcached-Session-Manager集群Session共享

    在本篇文章中,我们将深入探讨如何使用Nginx、Tomcat和Memcached-Session-Manager(MSM)来构建一个集群环境,并实现Session共享。这涉及到集群搭建、分布式应用以及缓存管理等多个方面的IT知识。下面将详细展开这些...

    nginx+tomcat+memcached负载均衡集群搭建许jar包大全

    2)Tomcat6和Tomcat7使用不同msm支持包:memcached-session-manager-tc6-1.6.5.jar和memcached-session-manager-tc7-1.6.5.jar,只可选一,否则启动报错。 3)msm源码中的lib包版本太低:spymemcached需要使用...

    tomcat7+nginx+memcached 配置tomcat 集群以及负载均衡

    通过以上步骤,我们可以构建出一个高效、可靠的基于Tomcat 7、Nginx和Memcached的Web应用集群,有效地解决了session共享问题,实现了负载均衡,提高了系统的稳定性和性能。在实际操作中,可能还需要根据具体需求调整...

    Nginx+Tomcat+Memcached集群 所需jar包集合

    couchbase-client-1.2.2.jar ...memcached-session-manager-1.6.5.jar memcached-session-manager-tc6-1.6.5.jar minlog-1.2.jar msm-kryo-serializer-1.6.5.jar reflectasm-0.9.jar spymemcached-2.10.2.jar

    nginx+tomcat+memcached

    在构建分布式系统时,Session共享是一个关键问题,特别是在使用多种服务器技术如Nginx、Tomcat和Memcached的环境中。本文将深入探讨如何利用这些组件实现跨服务器的Session共享,以确保用户在登录后能够在整个系统中...

    nginx+tomcat7+memcached所需jar

    通过这些jar包的集成,我们可以构建出一个高效的、基于Memcached的session管理解决方案,使得在nginx负载均衡下运行的Tomcat 7应用能充分利用Memcached进行session数据的存储和共享,从而提高系统的整体性能和可靠性...

    java基于nginx的tomcat负载均衡和集群

    Java 基于 Nginx 的 Tomcat 负载均衡和集群 在本文中,我们将讨论如何使用 Nginx 实现 Tomcat 的负载均衡和集群。这种方法可以解决 Session 共享的问题,并且可以使多台 Tomcat 服务器共享 Session。 首先,需要...

    Nginx+Tomcat+memcached实现session共享.txt

    memcached-session-manager-1.6.3.jar memcached-session-manager-tc6-1.6.3.jar minlog-1.2.jar msm-javolution-serializer-1.6.3.jar msm-kryo-serializer-1.6.3.jar msm-xstream-serializer-1.6.3.jar reflectasm...

    tomcat7+memcached+nginx 集群架包(无ngnix架包)

    【描述】中的"tomcat7+memcached session共享的架包"强调了这个集合主要关注的是在Tomcat7服务器上如何通过Memcached来实现session的跨服务器共享。在Web应用集群中,session共享是一个关键问题,因为用户在访问不同...

    Nginx+Tomcat6+Membercached 实现负载均衡

    标题中的“Nginx+Tomcat6+Memcached 实现负载均衡”涉及到的是一个常见的Web服务器和应用服务器的架构组合,用于提高系统性能和可扩展性。在这个架构中,Nginx作为反向代理和负载均衡器,Tomcat6是Java应用服务器,...

    nginx配置session+memcached所需jar包

    总结来说,实现Nginx、Tomcat集群与Memcached的Session共享,需要正确配置Nginx的反向代理设置,引入并配置Tomcat的Memcached Session Manager,以及确保所有必要的jar包已经正确安装。这一过程虽然涉及多个环节,但...

    nginx + 多个tomcat8 + memercached session 共享--所需的jar包

    常常会用到Nginx作为反向代理和负载均衡器,Tomcat作为应用服务器处理Java Web应用程序,而Memcached则作为一个分布式内存对象缓存系统,用于存储session数据,以实现多台服务器间的session共享。本篇文章将详细阐述...

    Arch-03-15- Nginx+tomcat 配置负载均衡动静分离

    标题“Arch-03-15- Nginx+tomcat 配置负载均衡动静分离”涉及的是在Web服务器架构中使用Nginx与Tomcat的集成,通过配置实现负载均衡和动静态资源分离。这样的架构可以提高系统的可用性和响应速度,减轻后端应用...

    Tomcat基于memcached的session保持-nginx反代

    通过上述步骤,我们实现了 Tomcat 集群中基于 Memcached 的 Session 保持功能,并利用 Nginx 作为反向代理来均衡负载。这种架构能够有效地解决分布式系统中的 Session 共享问题,提高系统的可扩展性和可用性。此外,...

    Nginx+tomcat集群的所有插件

    里面包括所有的包 ...memcached-session-manager-1.5.1.jar memcached-session-manager-tc7-1.5.1.jar msm-javolution-serializer-1.5.1.jar msm-kryo-serializer-1.5.1.jar msm-xstream-serializer-1.5.1.jar

Global site tag (gtag.js) - Google Analytics