Terry.Li-彬

虚其心,可解天下之问;专其心,可治天下之学;静其心,可悟天下之理;恒其心,可成天下之业。

  BlogJava :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 管理 ::
  143 随笔 :: 344 文章 :: 130 评论 :: 0 Trackbacks

1 概述

集群允许一个应用同时运行在多个并发的服务器。负载分布在不同的服务器,即使有个别服务器崩溃,应用程序还可以通过别的服务器节点访问。集群是企业级应用的关键技术,可以通过简单地增加节点来提升性能。

JBoss支持集群,启动一个JBoss集群的最简单的方式是,使用run -c all命令在局域网内启动多个JBoss实例。这些服务器实例,都使用all配置集,自动地构建一个集群。

1.1 定义

一个集群是由一系列节点组成。在一个JBoss集群中,一个节点是一个JBoss服务器实例,要构建一个集群,必须将几个JBoss服务器实例组合在一起。在一个局域网可能有多个集群,为了区分他们,每个集群必须有唯一的名称。

下图构建一个网络,包含三个集群。节点可以在任何时候添加到集群或者从集群中

Clusters and server nodes

注意:

一个JBoss服务器实例可以同时在多个集群,但是不推荐这样做,因为会增加管理复杂性。

JBoss服务器实例通过deploy/cluster-service.xml 文件中的ClusterPartition MBean指定属于哪一个集群。所有有同样的ClusterPartition MBean配置的节点属性同一个集群。因此,如果想在同一个局域网构建两个集群,只需要创建两个不同的ClusterPartition配置即可。如果某一个节点启动时,集群还不存在,会创建这个集群,同时当属于一个集群的所有节点移除时,集群也会被移除。

下面的示例是JBoss发布包的缺省定义,通过使用此缺省集群设置启动服务器,将可以构建一个缺省集群名称为DefaultPartition的集群。

  1. <mbean code="org.jboss.ha.framework.server.ClusterPartition"  
  2.     name="jboss:service=DefaultPartition">  
  3.             
  4.     >  
  5.     <attribute name="PartitionName">  
  6.         ${jboss.partition.name:DefaultPartition}   
  7.     attribute>  
  8.   
  9.     >  
  10.     <attribute name="NodeAddress">${jboss.bind.address}attribute>  
  11.   
  12.     >  
  13.     <attribute name="DeadlockDetection">Falseattribute>  
  14.         
  15.     
  16.         Increase for large states -->  
  17.     <attribute name="StateTransferTimeout">30000attribute>  
  18.   
  19.     >  
  20.     <attribute name="PartitionConfig">  
  21.         ... ...   
  22.     attribute>  
  23. mbean>  
xml 代码

这里省略了详细的JGroups协议配置,JGoups处理底层的节点之间的点对点通信,稍后将讨论其具体配置。ClusterPartition MBean可配置的属性包括:

  • PartitionName 缺省值是DefaultPartition,可选属性,用来指定集群的名称。

  • NodeAddress 设置节点的IP地址。

  • DeadlockDetection 缺省值为false,设置JGroups是否基于每个请求都运行消息死锁检测算法。

  • StateTransferTimeout 缺省值是30000,设置集群之间的状态复制的超时值。

  • PartitionConfig 指定JGourp配置选项。

一个集群包含的所有节点,必须有同样的 PartitionNameParitionConfig 元素。改变某些节点的这两个元素将导致集群分离,通过改变ParitionConfig 是更容易一些。

可以通过集群中的任何一个节点的JMX控制台查看当前集群信息(比如:http://hostname:8080/jmx-console),然后点击 jboss:service=DefaultPartition MBean,将可以看到当前集群的所有节点的IP地址列表。

注意

A cluster (partition) contains a set of nodes that work toward a same goal. Some clustering features require to sub-partition the cluster to achieve a better scalability. For example, let's imagine that we have a 10-node cluster and we want to replicate in memory the state of stateful session beans on all 10 different nodes to provide for fault-tolerant behaviour. It would mean that each node has to store a backup of the 9 other nodes. This would not scale at all (each node would need to carry the whole state cluster load). It is probably much better to have some kind of sub-partitions inside a cluster and have beans state exchanged only between nodes that are part of the same sub-partition. The future JBoss clustering implementation will support sub-partitions and it will allow the cluster administrator to determine the optimal size of a sub-partition. The sub-partition topology computation will be done dynamically by the cluster.

1.2 服务架构

对于应用程序开发人员来讲,可能更关心集群的架构,JBoss支持两种类型的集群架构,一种是客户端的拦截器架构,比如代理和存根,还有一种是负载平衡器。

1.2.1 客户端拦截器架构

JBoss应用服务器提供的大多数远程服务,包括JNDI,EJB,RMI等都要求客户端获取一个存根对象。存根对象由服务器生成,实现服务接口。客户端在本地调用存根对象,这些调用自动进行网络路由,激活服务器管理的服务对象。在一个集群环境中,服务器生成的存根对象也是一个拦截器实现,知道如何在集群包含的节点中路由请求。存根对象知道如何查找合适的服务器节点,编码调用参数,解码调用结果,并返回结果给客户端。

存根拦截器包含集群的相关信息,比如,所有可用的服务器节点的IP地址,负载平衡算法,以及当某个目标节点不可用时如何处理请求。每个请求中,服务器节点将用集群的最近的改变信息更新存根拦截器。比如,一个节点从集群中移除,当每个存根拦截器连接到任何一个激活的节点时,都将更新最新的配置信息。所有基于存根的操作对于客户来讲都是透明的,客户端拦截器集群架构如下图所示:

The client-side interceptor (proxy) architecture for clustering

1.2.2 负载平衡器

别的JBoss服务,特别是HTTP WEB服务,不要求客户端下载任何东西。客户直接通过某个通信协议发送请求并接受响应,比如HTTP协议。这种情况下,一个负载平衡器用来处理所有的请求并将这些请求派发给集群中的节点。负载平衡器通常是集群的一部分,它知道集群配置和恢复策略,客户端只需要知道负载平衡器。负载平衡器架构如下图所示:The load balancer architecture for clustering

负载平衡器架构的一个潜在的问题是负载平衡器是所有请求的唯一入口,必须保证高的可用性。

1.2.3 负载平衡策略

客户拦截器架构和负载平衡器架构都使用负载平衡策略确定由哪一个服务器节点发送一个新的请求。下面描述JBoss AS的负载平衡策略。

1.2.3.1 JBoss AS 3.0.x

JBoss 3.0.x 中有如下两个负载平衡选项可用:

  • Round-Robin (org.jboss.ha.framework.interfaces.RoundRobin):每次调用被派发给一个新的节点,第一个节点是随机选择的。

  • First Available (org.jboss.ha.framework.interfaces.FirstAvailable)::第一个可用的节点选为主节点,服务于随后每一个调用,主节点的选择是随机的。当集群中的节点发生变化时(因为节点启动或者停止),除非当前主节点依然可用,将重新选择一个可用的节点作为主节点。每个客户端拦截器或负载平衡器独立选择自己的主节点。

1.2.3.2 JBoss AS 3.2+

JBoss 3.2+ 中增加了一个新的负载平衡选项 "First AvailableIdenticalAllProxies" (org.jboss.ha.framework.interfaces.FirstAvailableIdenticalAllProxies),此选项和 "First Available" 策略有类似的策略,唯一的区别是选择的主节点被所有的客户端拦截器共享。

JBoss 3.2 以后的版本中, 定义了"Proxy Family"术语。is defined. A Proxy Family is a set of stub interceptors that all make invocations against the same replicated target. For EJBs for example, all stubs targeting the same EJB in a given cluster belong to the same proxy family. All interceptors of a given family share the same list of target nodes. Each interceptor also has the ability to share arbitrary information with other interceptors of the same family. A use case for the proxy family is give in Section 1.3.1, “Stateless Session Bean in EJB 2.x”.

1.2.4 布署

在一个集群中布署应用程序最容易的方式是使用farming服务,就是将一个应用程序压缩文件(EAR,WAR或SAR)热布署到任一个节点的 all/farm/ 目录,应用程序会自动地复制到集群中的其他节点。如果稍后有新的节点加入,新节点启动时也会自动从集群中复制应用程序并布署到本地。在集群运行的过程中,如果从任一节点的farm目录移除应用程序,这个应用程序将首先从本地卸载,然后再从集群的其他节点中卸载。应该手动从当前没有连接到集群的服务器节点中删除应用程序。

注意

当前存在一个Bug,Farm布署服务只能在热布署压缩文件的方式中才能生效。如果你首先放一个应用程序在farm目录,然后启动服务器,这个应用程序将不会被检测并布署到集群中。

注意

在farm目录中只能使用压缩文件,而不能使用展开的目录。

Farming 服务在JBoss AS发布版的all配置集中是缺省启用的,不需要手动设置。配置文件保存在deploy/deploy.last目录,如果需要在集群配置中启用farming服务,创建farm-service.xml文件如下,并拷贝到JBoss布署目录$JBOSS_HOME/server/your_own_config/deploy,确保在配置中已经启用集群服务。 

xml 代码
  1. xml version="1.0" encoding="UTF-8"?>       
  2. <server>           
  3.            
  4.     <mbean code="org.jboss.ha.framework.server.FarmMemberService"        
  5.             name="jboss:service=FarmMember,partition=DefaultPartition">        
  6.         ...         
  7.         <attribute name="PartitionName">DefaultPartitionattribute>         
  8.         <attribute name="ScanPeriod">5000attribute>         
  9.         <attribute name="URLs">farm/attribute>        
  10.     mbean>          
  11. server>  
 

  FarmMemberService MBean 可用的属性如下:

  • PartitionName 指定集群名字,缺省值为 DefaultPartition.

  • URLs 指定布署器监控布署文件的目录,如果不存在,MBean将创建。 "." 表示配置目录(比如 $JBOSS_HOME/server/all/)。

  • ScanPeriod 指定文件夹的扫描周期,缺省值为5000。

Farming 服务是URLDeploymentScanner的扩展,用来扫描deploy目录,因此在FarmMemberService MBean中可以使用URLDeploymentScanner MBean 中定义的所有属性。

1.2.4 分布式状态复制服务

集群环境中,分布式状态管理是集群必须提供的一个关键服务。比如一个有状态的会话Bean应用程序,会话状态必须在所有节点的Bean实例中保持同步,因此不管哪个节点服务请求,客户端应该得到同样的会话状态。实体Bean应用程序中,Bean实例必须在集群中进行缓存以减轻数据库负载。目前JBoss AS中的状态复制和分布式缓存由两种方式提供: HASessionState MBean 和 JBoss Cache 框架。

  • HASessionState MBean 基于EJB2.X有状态会话Bean和JBoss3.x和4.x的HTTP负载平衡器提供会话复制和分布式缓存服务。MBean相关定义在 all/deploy/cluster-service.xml 文件中。

  • JBoss Cache 是一个分布式缓存框架,可以用在任何应用程序服务器环境,也可以独立使用,已经逐渐替换 HASessionState 服务。JBoss AS 整合JBoss Cache 为HTTP会话, EJB 3.0 会话和实体Bean,Hibernate持久对象提供缓存服务。这些缓存服务都在独立的MBean中定义。

1.5 HTTP服务

HTTP会话复制用来在你的WEB客户端和集群的节点之间复制状态,那么当某一个节点崩溃时,集群中的其他节点可以恢复。要实现这个目的,必须执行两个步骤:

  • 会话状态复制

  • 请求负载平衡

如果使用all配置集启动JBoss,缺省会启用会话状态复制功能,只需要布署你的应用程序,会话状态就已经可以在集群中进行复制。

但是负载平衡需要额外的软件支持,作为一个常用的场景,我们将描述如何设置Apache和mod_jk。这个功能也可以通过专门的硬件或者其他软件来实现。

注意

一个负载平衡器跟踪HTTP请求,根据和请求关联的会话,决定派发请求到合适的节点。这被称作使用sticky-sessions的负载平衡器:一 旦在某个节点创建了会话,随后的请求将由同一个节点进行处理。使用一个支持sticky-sessions的负载平衡器可以不用复制会话,没有会话状态复 制的花销,每个请求将是被同一个节点处理。但是一旦这个节点崩溃,保存在这个节点的所有客户会话都将丢失,客户必须登录进另一个节点,重新启动一个新的会 话。在某些情况,不复制HTTP会话是可以接受的,因为比较关键的状态都保存在数据库中,但是也有些情况下,丢失会话状态是无法接受的,这种情况下,会话 状态复制的花销是必须承受的。

Apache 是一个可以通过插入模块扩展功能的WEB服务器,mod_jk(最新的mod_jk2)模块的主要功能就是允许从Apache分发请求到一个 Servlet容器,还可以支持在多个Servlet容器中负载平衡HTTP调用(遵循sticky-sessions)。

1.5.1. 下载软件

http://httpd.apache.org/ 下载Apache,安装是非常直白的,不需要特别的配置,推荐使用2.0.x版本,这里假定Apache安装目录为APACHE_HOME。 

然后从www.apache.org/dist/jakarta/tomcat-connectors/jk/binaries/. 下载mod_jk模块,建议使用mod_jk 1.2.x版本。重命名下载的文件为mod_jk.so,拷贝到APACHE_HOME/modules/目录下。

1.5.2. 配置Apache使用mod_jk

更改APACHE_HOME/conf/httpd.conf 文件,在文件尾部添加一行:

java 代码
  1. # Include mod_jk's specific configuration file     
  2. Include conf/mod-jk.conf    

创建文件APACHE_HOME/conf/mod-jk.conf:

java 代码
  1. # Load mod_jk module   
  2. # Specify the filename of the mod_jk lib   
  3. LoadModule jk_module modules/mod_jk.so   
  4.     
  5. # Where to find workers.properties   
  6. JkWorkersFile conf/workers.properties   
  7.   
  8. # Where to put jk logs   
  9. JkLogFile logs/mod_jk.log   
  10.     
  11. # Set the jk log level [debug/error/info]   
  12. JkLogLevel info    
  13.     
  14. # Select the log format   
  15. JkLogStampFormat  "[%a %b %d %H:%M:%S %Y]"  
  16.     
  17. # JkOptions indicates to send SSK KEY SIZE   
  18. JkOptions +ForwardKeySize +ForwardURICompat -ForwardDirectories   
  19.     
  20. # JkRequestLogFormat   
  21. JkRequestLogFormat "%w %V %T"  
  22.                   
  23. # Mount your applications   
  24. JkMount /application/* loadbalancer   
  25.     
  26. # You can use external file for mount points.   
  27. # It will be checked for updates each 60 seconds.   
  28. # The format of the file is: /url=worker   
  29. # /examples/*=loadbalancer   
  30. JkMountFile conf/uriworkermap.properties                  
  31.   
  32. # Add shared memory.   
  33. # This directive is present with 1.2.10 and   
  34. # later versions of mod_jk, and is needed for  
  35. # for load balancing to work properly   
  36. JkShmFile logs/jk.shm    
  37.                  
  38. # Add jkstatus for managing runtime data   
  39.   
  40.     JkMount status   
  41.     Order deny,allow   
  42.     Deny from all   
  43.     Allow from 127.0.0.1  
  44.    

其中有两个设置是非常重要的:

  • LoadModule 指令必须引用下载的mod_jk库,路径中必须包括modules前缀。

  • JkMount 指令通知Apache哪些URL应该被派发给mod_jk模块(也就是派发给Servlet容器)。上面的配置中,所有基于/application/* 的请求都会派发给mod_jk负载平衡器。如果使用mod_jk作为一个专门的负载平衡器,可以将所有的请求(/*)派发给mod_jk。

除了使用 JkMount 指令,也可以使用JkMountFile 指令指定一个配置文件,包含多个URL映射配置。你只需要在APACHE_HOME/conf 目录下创建一个uriworkermap.properties 文件。文件格式为/url=worker_name,示例文件如下:

java 代码
  1. # Simple worker configuration file   
  2.   
  3. # Mount the Servlet context to the ajp13 worker   
  4. /jmx-console=loadbalancer   
  5. /jmx-console/*=loadbalancer   
  6. /web-console=loadbalancer   
  7. /web-console/*=loadbalancer   

如上示例将配置 mod_jk 派发请求 /jmx-console/web-console 给Tomcat。

1.5.3. 在mod_jk中配置工作者节点

接着需要创建conf/workers.properties文件,这个文件用来指定名个不同的Servlet容器的位置以及调用如何在这些容器之间进行平衡。针对一个两个节点的设置,示例文件如下:

java 代码
  1. # Define list of workers that will be used   
  2. # for mapping requests   
  3. worker.list=loadbalancer,status   
  4.   
  5. # Define Node1   
  6. # modify the host as your host IP or DNS name.   
  7. worker.node1.port=8009  
  8. worker.node1.host=node1.mydomain.com    
  9. worker.node1.type=ajp13   
  10. worker.node1.lbfactor=1  
  11. worker.node1.cachesize=10  
  12.   
  13. # Define Node2   
  14. # modify the host as your host IP or DNS name.   
  15. worker.node2.port=8009  
  16. worker.node2.host= node2.mydomain.com   
  17. worker.node2.type=ajp13   
  18. worker.node2.lbfactor=1  
  19. worker.node2.cachesize=10  
  20.   
  21. # Load-balancing behaviour   
  22. worker.loadbalancer.type=lb   
  23. worker.loadbalancer.balance_workers=node1,node2   
  24. worker.loadbalancer.sticky_session=1  
  25. #worker.list=loadbalancer   
  26.   
  27. # Status worker for managing load balancer   
  28. worker.status.type=status              

上述文件配置mod_jk执行Round-Robin的基于sticky-sessions的负载平衡策略,两个节点都监听8009端口。

works.properties文件中,每个节点使用worker.XXX 命名规范进行定义,这里 XXX 可以是任何名字,用来命名Servlet容器。对于每个工作者,必须指定在目标Servlet容器中配置的AJP13连接器的IP地址和端口号。

lbfactor 属性用来表示负载平衡权重,决定节点之间的优先级,值越大,该节点将接受越多的请求。这个选项可以用来分配不同的节点不同的负载。

cachesize 属性定义关联的Servlet容器的线程池大小,确定这个值没有超过Servlet容器在AJP13连接器中的配置。可以参考jakarta.apache.org/tomcat/connectors-doc/config/workers.html

conf/workers.properties 最后定义负载平衡工作者,唯一需要改变的是worker.loadbalancer.balanced_workers 设置,必须列出刚才定义的所有工作者。

sticky_session 属性指定集群针对HTTP会话的处理方式,如果指定worker.loadbalancer.sticky_session=0,每个请求将在两个节点中进行负载平衡。但是当一个用户在某个节点创建一个会话时,比较好的方式是随后的请求都导向这个节点,这被称作"sticky session",由于客户总是使用会话创建的节点服务所有随后的请求。否则用户的会话数据必须在两个节点进行同步,要启用"sticky session",必须设置 worker.loadbalancer.sticky_session 值为1。

注意

A non-loadbalanced setup with a single node required the worker.list=node1 entry before mod_jk would function correctly.

1.5.4. 配置JBoss

最后必须配置集群中所有节点的JBoss Tomcat实例。

在每个节点中,我们必须根据workers.properties文件中的worker名称来命名节点。比如,在JBoss 实例node1中,需要编辑JBOSS_HOME/server/all/deploy/jbossweb-tomcat50.sar/server.xml 文件,定位 元素添加一个jvmRoute属性:

xml 代码
  1. <Engine name="jboss.web" defaultHost="localhost" jvmRoute="node1">  
  2. ... ...   
  3. Engine>  

然后,对于集群中的每一个节点,我们必须通知它添加一个jvmRoute值到会话cookies中,以便mod_jk可以路由随后的请求。编辑 JBOSS_HOME/server/all/deploy/jbossweb-tomcat50.sar/META-INF/jboss-service.xml 文件,定义UseJK的元素,设置值为true:

xml 代码
  1. <attribute name="UseJK">trueattribute>  

到此为止,我们已经成功设置Apache_mod_jk使用sticky-session方式的负载平衡。

1.5.5. 配置HTTP会话状态复制

目前为止,已经成功配置了sticky-session方式的负载平衡,但是这种方式并不是一个完善的解决方案,因为一个节点崩溃,所有的会话数据都会丢失。更可选的解决方式是在集群的节点之间复制会话状态,这样客户访问任一个节点都会获得同样的会话状态。

jboss.cache:service=TomcatClusteringCache MBean 使用 JBoss Cache提供HTTP会话状态复制服务,这个MBean定义在deploy/tc5-cluster.sar/META-INF/jboss-service.xml 文件中。

注意

JBoss AS 4.0.4 CR2以前的版本,HTTP会话缓存配置文件在deploy/tc5-cluster-service.xml文件中。

下面是一个典型的deploy/tc5-cluster.sar/META-INF/jboss-service.xml 文件

xml 代码
  1. <mbean code="org.jboss.cache.aop.TreeCacheAop"  
  2.     name="jboss.cache:service=TomcatClusteringCache">  
  3.   
  4.     <depends>jboss:service=Namingdepends>  
  5.     <depends>jboss:service=TransactionManagerdepends>  
  6.     <depends>jboss.aop:service=AspectDeployerdepends>  
  7.   
  8.     <attribute name="TransactionManagerLookupClass">  
  9.         org.jboss.cache.BatchModeTransactionManagerLookup   
  10.     attribute>  
  11.        
  12.     <attribute name="IsolationLevel">REPEATABLE_READattribute>  
  13.        
  14.     <attribute name="CacheMode">REPL_ASYNCattribute>  
  15.        
  16.     <attribute name="ClusterName">  
  17.       Tomcat-${jboss.partition.name:Cluster}   
  18.     attribute>  
  19.        
  20.     <attribute name="UseMarshalling">falseattribute>  
  21.        
  22.     <attribute name="InactiveOnStartup">falseattribute>  
  23.        
  24.     <attribute name="ClusterConfig">  
  25.         ... ...   
  26.     attribute>  
  27.        
  28.     <attribute name="LockAcquisitionTimeout">15000attribute>  
  29. mbean>  
 

详细的配置请参见JBossCache缓存配置部分,下面讨论几个比较关键的配置:

  • TransactionManagerLookupClass 设置事务管理器工厂,缺省值为org.jboss.cache.BatchModeTransactionManagerLookup,这个类通知缓存不要参与JTA事务,自己管理事务。

  • IsolationLevel 设置更新分布式缓存的隔离级别,可选值包括:SERIALIZABLE, REPEATABLE_READ, READ_COMMITTED, READ_UNCOMMITTED, 和 NONE。 这里的隔离级别和数据库的隔离级别有同样的含义,对于大多数WEB应用程序来讲通常设置为REPEATABLE_READ

  • CacheMode 控制缓存如何被复制。可选值包括:REPL_SYNCREPL_ASYNC,确定改变是应该同步还是异步复制。使用同步复制,确保在请求完成之前传播改变,同步复制相对来说会慢一些。

  • ClusterName 指定集群名称。缺省的集群名称是由当前的JBoss分区名称加上"Tomcat-"前缀。所有的节点应该使用相同的集群名称。

  • UseMarshallingInactiveOnStartup 属性必须有相同的值,如果使用FIELD级别的会话复制,这两个值必须为true。

  • ClusterConfig 配置底层的JGoups 堆栈。最重要的配置元素是广播地址和端口号mcast_addrmcast_port,详细配置请参见JGroups配置。

  • LockAcquisitionTimeout 设置获取锁的最大超时值,缺省值为15000。

  • UseReplQueue 使用异步复制时是否启动复制队列,这允许多个更新一起执行从而提升性能。复制队列属性由 ReplQueueIntervalReplQueueMaxElements 属性配置。

  • ReplQueueInterval 配置JBoss缓存处理复制队列的时间间隔。

  • ReplQueueMaxElements: 配置复制队列可以保存的最多的元素数目。

1.5.6. 在应用程序中启用会话复制

要在你的WEB应用程序中启用集群,必须在web.xml文件中声明distributable。示例如下:

xml 代码
  1.   
  2. web-app>  
 
可以在jboss-web.xml文件中进一步配置会话复制,示例如下:

xml 代码
  1. <jboss-web>  
  2.     <replication-config>  
  3.         <replication-trigger>SET_AND_NON_PRIMITIVE_GETreplication-trigger>  
  4.         <replication-granularity>SESSIONreplication-granularity>  
  5.         <replication-field-batch-mode>truereplication-field-batch-mode>  
  6.     replication-config>  
  7. jboss-web>  

replication-trigger 元素确定什么情况触发一次会话复制,有四个选项:

  • SET: 使用此选项,一个会话只有在设置属性时才被认为需要复制。如果你的应用总是需要改变会话的属性,这个选项将是性能最优的。如果一个对象只是被检索和修改,但是不需要改写会话,这个对象的改变并不会造成会话复制。

  • SET_AND_GET: 使用此选项,任何属性的获取和设置都被认为需要复制。如果一个对象被检索和修改,但是不需要改定会话,这个对象的改变将会被复制。这个选项有显著的性能问题。

  • SET_AND_NON_PRIMITIVE_GET: 此选项类似于 SET_AND_GET,唯一的例外是只有非原始类型的检索操作被认为需要复制。比如,HTTP会话请求可以从属性中检索一个非原始类型的对象实例并更改,如果我们没有指定此选项,更改将不会被复制。这是缺省设置。

  • ACCESS: 使用此选项,只要会话被访问都被认为需要复制。由于每个HTTP请求都会造成会话被访问,所以每个请求都会导致会话复制。使用此选项,会话时间戳将在集群中同步。注意使用此选项会有显著的性能问题。

replication-granularity 元素控制复制粒度,支持的选项包括:

  • SESSION: 基于会话的复制,只要会话发生改变,整个会话对象都会被序列化。

  • ATTRIBUTE: 基于属性的复制,复制只在属性被认为是脏的时候发生,比如lastAccessTime.。对于持有大量数据的会话,这个选项可以提升性能。

  • FIELD: 基于字段的复制,复制只在会话属性包含的对象的字段发生变化时发生。

replication-field-batch-mode 表示是否需要在每个HTTP请求进行批量更新,缺省值为true。

如果你的会话通常比较小,SESSION选项比较适合,如果会话比较大,而且某些部分不是经常被访问,ATTRIBUTE 选项比较适合,如果会话属性包含大数据量的对象而且只有字段经常更改,FIELD 选项比较适合。

1.5.7. 使用FIELD级别的复制

FIELD-level replication only replicates modified data fields inside objects stored in the session. It could potentially drastically reduce the data traffic between clustered nodes, and hence improve the performance of the whole cluster. To use FIELD-level replication, you have to first prepare your Java class to indicate which fields are to be replicated. This is done via JDK 1.4 style annotations embedded in JavaDocs:

To annotate your POJO, we provide two annotations: @@org.jboss.web.tomcat.tc5.session.AopMarker and @@org.jboss.web.tomcat.tc5.session.InstanceAopMarker. When you annotate your class with AopMarker, you indicate that instances of this class will be used in FIELD-level replication. For exmaple,

/*
* My usual comments here first.
* @@org.jboss.web.tomcat.tc5.session.AopMarker
*/
public class Address
{
...
}

If you annotate it with InstanceAopMarker instead, then all of its sub-class will be automatically annotated as well. For example,

/*
*
* @@org.jboss.web.tomcat.tc5.session.InstanceOfAopMarker
*/
public class Person
{
...
}

then when you have a sub-class like

public class Student extends Person
{
...
}

there will be no need to annotate Student. It will be annotated automatically because it is a sub-class of Person.

However, since we only support JDK 1.4 style annotation (provided by JBoss Aop) now, you will need to perform a pre-processing step. You need to use the JBoss AOP pre-compiler annotationc and post-compiler aopc to process the above source code before and after they are compiled by the Java compiler. Here is an example on how to invoke those commands from command line.

$ annotationc [classpath] [source files or directories]
$ javac -cp [classpath] [source files or directories]
$ aopc [classpath] [class files or directories]

Please see the JBoss AOP documentation for the usage of the pre- and post-compiler. The JBoss AOP project also provides easy to use ANT tasks to help integrate those steps into your application build process. In the next AS release, JDK 5.0 annotation support will be provided for greater transparency. But for now, it is important that you perform the pre- and post-compilation steps for your source code.

Note

Or, you can see a complete example on how to build, deploy, and validate a FIELD-level replicated web application from this page: http://wiki.jboss.org/wiki/Wiki.jsp?page=Http_session_field_level_example. The example bundles the pre- and post-compile tools so you do not need to download JBoss AOP separately.

When you deploy the web application into JBoss AS, make sure that the following configurations are correct:

  • In the server's deploy/tc5-cluster.sar/META-INF/jboss-service.xml file, the inactiveOnStartup and useMarshalling attributes must both be true.

  • In the application's jboss-web.xml file, the replication-granularity attribute must be FIELD.

Finally, let's see an example on how to use FIELD-level replication on those data classes. Notice that there is no need to call session.setAttribute() after you make changes to the data object, and all changes to the fields are automatically replicated across the cluster.

<pre class="programlistin
posted on 2008-08-19 11:18 礼物 阅读(3653) 评论(0)  编辑  收藏 所属分类: JBoss