Tomcat7 自带的集群功能是通过session复制完成的,现有两个复制方式:
- DeltaManager:
将session复制到所有tomcat节点中,不管是否有相应的应用(it will replicate to all nodes, even nodes that don't have the application deployed.)。缺点是如果集群节点很多,此种方式的系统消耗太大。具体节点多少算多,官方没有给出数量级。
- BackupManager:
将session复制到一个备份节点中,并且只复制到有相应应用的节点。(This manager only replicates the session data to one backup node, and only to nodes that have the application deployed.)缺点是这种方式没有像DeltaManager一样经过大规模的应用测试。
一个简单的Tomcat集群结构
DNS Round Robin
|
Load Balancer
/
Cluster1 Cluster2
/ /
Tomcat1 Tomcat2 Tomcat3 Tomcat4
Tomcat7中运行session复制,需要完成以下步骤(先有个印象,再看下面的实例):
- session中所有的属性(attributes)必须实现java.io.Serializable
- 去掉配置文件server.xml中cluster元素的注释
- 如果要自定义cluster valves,确认在Cluster元素里定义ReplicationValve。
- 如果Tomcat节点运行在同一台服务器上,确保tcpListenPort属性值唯一。在大多数情况下,Tomcat可以自动检测4000-4100范围内的端口
- 确保应用中web.xml文件有<distributable/>元素
- 如果使用mod_jk,确保设置jvmRoute属性<Engine name="Catalina" jvmRoute="node01">,并且这个属性要和workers.properties相同
- 确保所有节点都有相同的时间。
- 确保loadbalancer被设置为粘性session模式(sticky session)
注意:浏览器会在某些情况下创建新的Session。
配置:
<Cluster className="org.apache.catalina.ha.tcp.SimpleTcpCluster" channelSendOptions="8"> <!-- <Manager className="org.apache.catalina.ha.session.BackupManager" expireSessionsOnShutdown="false" notifyListenersOnReplication="true" mapSendOptions="6"/> --> <Manager className="org.apache.catalina.ha.session.DeltaManager" expireSessionsOnShutdown="false" notifyListenersOnReplication="true"/> <Channel className="org.apache.catalina.tribes.group.GroupChannel"> <Membership className="org.apache.catalina.tribes.membership.McastService" address="228.0.0.4" port="45564" frequency="500" dropTime="3000"/> <Receiver className="org.apache.catalina.tribes.transport.nio.NioReceiver" address="auto" port="5000" selectorTimeout="100" maxThreads="6"/> <Sender className="org.apache.catalina.tribes.transport.ReplicationTransmitter"> <Transport className="org.apache.catalina.tribes.transport.nio.PooledParallelSender"/> </Sender> <Interceptor className="org.apache.catalina.tribes.group.interceptors.TcpFailureDetector"/> <Interceptor className="org.apache.catalina.tribes.group.interceptors.MessageDispatch15Interceptor"/> <Interceptor className="org.apache.catalina.tribes.group.interceptors.ThroughputInterceptor"/> </Channel> <Valve className="org.apache.catalina.ha.tcp.ReplicationValve" filter=".*.gif|.*.js|.*.jpeg|.*.jpg|.*.png|.*.htm|.*.html|.*.css|.*.txt"/> <Deployer className="org.apache.catalina.ha.deploy.FarmWarDeployer" tempDir="/tmp/war-temp/" deployDir="/tmp/war-deploy/" watchDir="/tmp/war-listen/" watchEnabled="false"/> <ClusterListener className="org.apache.catalina.ha.session.ClusterSessionListener"/> </Cluster>
下面详细说明一下:
<Cluster className="org.apache.catalina.ha.tcp.SimpleTcpCluster" channelSendOptions="6">
Cluster元素可以是<Engine>或者<Host>的子元素,如果使用farm war deploy功能,则必须是<Host>的子元素。SimpleTcpCluster是官方提供的唯一一个Cluster。channaelSendOptions是设置SimpleTcpCluster.send方法发送消息的方式,默认值是8。参考:http://tomcat.apache.org/tomcat-7.0-doc/api/org/apache/catalina/tribes/Channel.html
TcpFailureDetector - 通过TCP验证成员是否正常。
MessageDispatch15Interceptor - 调度消息到线程(线程池),异步发送消息。
ThroughputInterceptor - 打印出消息流量的简单统计
注意interceptor的配置顺序。
- <Deployer className="org.apache.catalina.ha.deploy.FarmWarDeployer"
- tempDir="/tmp/war-temp/"
- deployDir="/tmp/war-deploy/"
- watchDir="/tmp/war-listen/"
- watchEnabled="false"/>
当使用DeltaManager时,通过ClusterSessionListener接收和传播信息。
三、测试
到现在为止我们配置了一个Tomcat,复制这个Tomcat,修改应该修改的部分(如果是同一台机器则需要修改一些端口和jvmRoute,如果不是同一台机器只需修改jvmRoute)。
之后建立一个简单的webapp,记得web.xml中添加<distributable/>元素。如下:
![](https://images0.cnblogs.com/i/457851/201403/031856087619478.png)
![](https://images0.cnblogs.com/i/457851/201403/031856254947091.png)
访问两个tomcat 发现sessionId相同 说明session复制成功!
参考:http://toplchx.iteye.com/blog/1924322