一人拾柴火不旺,众人拾柴火焰高。Tomcat服务器也是一样,一台服务器再强大能承受的访问也是有限的。要提供高并发、高可用的服务,就必须横向扩展,多台Tomcat组成一个集群,根据实际的访问量动态增减服务器的部署。

负载均衡的难点

我们一般用session来保持会话,所以Tomcat服务器是有状态的。坏处是当一台宕机后自动跳转到另一台服务器可能会导致用户会话失效,最明显的例子就是要重新登录。所以以下的几种方法,都是在围绕session的问题。

方案一、使用Tomcat自带的session同步功能

如果我们使用nginx作为负载均衡服务器,它默认使用是轮询策略(也就是第一次请求分给服务器A,第二次分配给B,以此类推)。这种方案非常简单,但是别忘了我们的session是放在访问服务器上,轮询得结果是下次我们可能访问的是另一台服务器了,导致我们的会话不能保持。
其实,解决起来也似乎很简单,就是创建session时,只需要把它复制给集群中的所有服务器就行了,下次不管访问哪一台Tomcat,它都能根据sessionID找到我们的session。非常幸运的是,Tomcat本身就已经帮我们实现了这个功能——session复制。我们只需简单两步配置即可:
  1. 在server.xml中打开以下注释:

    <Cluster className="org.apache.catalina.ha.tcp.SimpleTcpCluster"/>
  2. 在tomcat或者应用下web.xml,添加:

    <distributable/>

    是不是非常简单实用呢。

    简单归简单,不得不提的是,它有一个缺点,因为它默认使用BackupManager,是all-to-all的复制,所以如果大量节点的集群会有广播风暴,而且即使没有部署应用的节点也会复制。所以对大规模的集群,这种方案需谨慎考虑哦。

方案二、使用nginx的ip_hash策略

上面说了,nginx默认实用的是轮询策略,所以得配合session复制功能,才能保持会话。其实我们有一个更简单的方法,实用nginx的ip_hash策略。顾名思义,就是对ip进行hash散列,ip散列后找到指定的服务器,下次就一直访问这台服务器,这样一来,就不需要session复制了。
配置起来也非常简单,nginx自身支持,只需在nginx配置中添加配置即可,Tomcat不需要任何处理:
upstream mywebsite {
    server 110.119.88.1:8080;
    server 110.119.88.1:8080;
    ip_hash;
}

凡事都有两面,它简单但也有一些缺点:

  1. nginx必须作为最前端的服务器,否则得不到正确的ip(自己暂未证实)
  2. nginx后不能再有其他的负载均衡,否则不能保证每次访问的是同一台服务器(未证实,不过按理应该是这样)
  3. 同一局域网下的多个用户可能每次访问的都是同一服务器。如我们的应用是给一个单位使用,使用ip_hash分流会很不理想
  4. 如果一台服务器挂了之后,自动跳转到另一台服务器,没有session,会需要重新登录

方案三、使用nginx的nginx_upstream_jvm_route扩展模块

上面两种方案都是nginx已支持的策略,接下来这种需要在nginx安装扩展模块。配置起来稍微复杂一些,需要完成以下三个步骤:
  1. 为nginx添加扩展模块
  2. Tomcat的server.xml标注jvmRoute

    <Engine name="Catalina" defaultHost="localhost" jvmRoute="a">
  3. 配置文件nginx.conf中指明负载策略

    jvm_route $cookie_JSESSIONID;

    需要注意的是,这个模块通过session和cookie实现会话保持,如果url没有添加sessionid,或者用户禁用了浏览器cookie,则使用nginx默认的轮询,也就不会保持会话了。

    同样,他也有很大的缺点:我在配置实用时,没有找到支持nginx最新版本jvm_route,只找到支持nginx1.6的模块。

方案四、使用Memcached或Redis统一管理session

和上面的比起来,这个方案在大规模集群中更合理,管理起来更方便。
使用Memcached或者redis集中管理session,而不是让Tomcat管理,所以每个Tomcat都是对等的、无状态的,集群便可以随意增减Tomcat了。使用的Memcached_Session_Manager管理session,安装Memcached服务器后,配置起来比较简单,就是在tomcat的context.xml配置中,指定使用Memcached管理器来管理session。
<Manager className="de.javakaffee.web.msm.MemcachedBackupSessionManager"
        memcachedNodes="n1:memcached_server_ip:11211"
        requestUriIgnorePattern=".*\.(png|gif|jpg|css|js)$"
        sessionBackupAsync="false"
        sticky="false"
        sessionBackupTimeout="100"
        transcoderFactoryClass="de.javakaffee.web.msm.serializer.javolution.JavolutionTranscoderFactory"
        copyCollectionsForSerialization="false" />

总结

没有最好的,只有最适合的,综合实际情况,选择最适合当前项目的方案才是最好的方案。

众人拾柴火焰高之Tomcat集群的更多相关文章

  1. 整合apache+tomcat+keepalived实现高可用tomcat集群

    Apache是一个强大的Web服务器在处理静态页面.处理大量网络客户请求.支持服务的种类以及可配置方面都有优势,高速并且强壮.但是没有JSP/Servlet的解析能力.整合Apache和Tomcat可 ...

  2. Keepalived+Nginx+Tomcat 实现高可用Web集群

    https://www.jianshu.com/p/bc34f9101c5e Keepalived+Nginx+Tomcat 实现高可用Web集群 0.3912018.01.08 20:28:59字数 ...

  3. 【原创】Tomcat集群环境下对session进行外部缓存的方法(2)

    Session对象的持久化比较麻烦,虽然有序列化,但是并不确定Session对象中保存的其他信息是否可以序列化,这可能是网上很多解决方案摒弃此种做法的原因,网上的很多做法都是将Session中的att ...

  4. Tomcat集群配置学习篇-----分布式应用

    Tomcat集群配置学习篇-----分布式应用 现目前基于javaWeb开发的应用系统已经比比皆是,尤其是电子商务网站,要想网站发展壮大,那么必然就得能够承受住庞大的网站访问量:大家知道如果服务器访问 ...

  5. Linux+Apache+Tomcat集群配置

    参考: http://blog.csdn.net/bluishglc/article/details/6867358# http://andashu.blog.51cto.com/8673810/13 ...

  6. linux+apache+mod_Jk+tomcat实现tomcat集群

    最近一段时间一直在研究实现apache + jk_mod + tomcat实现负载均衡,起初负载均衡算是配置蛮顺利的,但是到了配置tomcat集群时所有配置都没有问题,但是tomcat日志中一直提示没 ...

  7. apache、mod_jk负载均衡与tomcat集群

    最近需要搭建apache和tomcat的集群,实现静态网站直接通过apache访问,动态网站转交给tomcat处理,实现负载均衡和tomcat集群配置. apache安装 wget http://ap ...

  8. 【转】Tomcat集群Cluster实现原理剖析

    此文章来源:http://zyycaesar.iteye.com/blog/296606 此文章作者:zyycaesar 对于WEB应用集群的技术实现而言,最大的难点就是如何能在集群中的多个节点之间保 ...

  9. Linux下搭建tomcat集群全记录(转)

    本文将讲述如何在Linux下搭建tomcat集群,以及搭建过程中可能的遇到的问题和解决方法.为简单起见,本文演示搭建的集群只有两个tomact节点外加一个apache组成,三者将安装在同一机器上:ap ...

随机推荐

  1. (中等) HDU 3335 , DLX+重复覆盖。

    Description As we know,the fzu AekdyCoin is famous of math,especially in the field of number theory. ...

  2. android测试之——Instrumentation(一)

    以下是本人原创,如若转载和使用请注明转载地址.本博客信息切勿用于商业,可以个人使用,若喜欢我的博客,请关注我,谢谢!博客地址 感谢您支持我的博客,我的动力是您的支持和关注!如若转载和使用请注明转载地址 ...

  3. Spring MVC 关于分页的简单实现

    据本人了解,目前较常用的分页实现办法有两种: 1.每次翻页都修改SQL,向SQL传入相关参数去数据库实时查出该页的数据并显示. 2.查出数据库某张表的全部数据,再通过在业务逻辑里面进行处理去取得某些数 ...

  4. C语言-if语句

    1.一般形式 if(表达式)表达式1 else 表达式2   :表达式成立(为真)则执行表达式1,否则执行表达式2. 适用范围:真假,对错,开关,对立面的条件 注意:如果if语句中只包括一条语句,可以 ...

  5. Oracle数据库中的函数

    1.随机数函数:DBMS_RANDOM.RANDOM )) FROM DUAL; --产生一个100以内的随机数 *dbms_random.value) FROM dual; --产生一个100-10 ...

  6. bzoj1070————2016——3——14

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=1070: 题目概括: Description 同一时刻有N位车主带着他们的爱车来到了汽车维修中 ...

  7. Hao Yin Jian 寒假第一周

    题目链接:https://vjudge.net/contest/147561#problem/A 题意:除法运算,abcde / fghij = n,从小到大输出,其中abcdefghij为0~9的不 ...

  8. Mysql 多列形成主键(复合主键 )

    什么是数据表的复合主键 所谓的复合主键 就是指你表的主键含有一个以上的字段组成 比如 create table test (    name varchar(19),    id number,    ...

  9. iOS 之 导航栏按钮

    UIButton *releaseButton = [UIButton buttonWithType:UIButtonTypeRoundedRect]; [releaseButton setTitle ...

  10. Leetcode 175. Combine Two Tables

    Table: Person +-------------+---------+ | Column Name | Type | +-------------+---------+ | PersonId ...