1 概述
集群允许一个应用同时运行在多个并发的服务器。负载分布在不同的服务器,即使有个别服务器崩溃,应用程序还可以通过别的服务器节点访问。集群是企业级应用的关键技术,可以通过简单地增加节点来提升性能。
JBoss支持集群,启动一个JBoss集群的最简单的方式是,使用run -c all命令在局域网内启动多个JBoss实例。这些服务器实例,都使用all配置集,自动地构建一个集群。
1.1 定义
一个集群是由一系列节点组成。在一个JBoss集群中,一个节点是一个JBoss服务器实例,要构建一个集群,必须将几个JBoss服务器实例组合在一起。在一个局域网可能有多个集群,为了区分他们,每个集群必须有唯一的名称。
下图构建一个网络,包含三个集群。节点可以在任何时候添加到集群或者从集群中
注意:
一个JBoss服务器实例可以同时在多个集群,但是不推荐这样做,因为会增加管理复杂性。
JBoss服务器实例通过deploy/cluster-service.xml
下面的示例是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. <!--- Max time (in ms) to wait for state transfer to complete. </sp-->
16. >
17. <attribute name="StateTransferTimeout">30000attribute>
18.
19. >
20. <attribute name="PartitionConfig">
21. ... ...
22. attribute>
23. mbean>
xml 代码
这里省略了详细的JGroups协议配置,JGoups处理底层的节点之间的点对点通信,稍后将讨论其具体配置。ClusterPartition
- PartitionName 缺省值是DefaultPartition,可选属性,用来指定集群的名称。
- NodeAddress
- DeadlockDetection
- StateTransferTimeout
- PartitionConfig
一个集群包含的所有节点,必须有同样的 PartitionName 和 ParitionConfig 元素。改变某些节点的这两个元素将导致集群分离,通过改变ParitionConfig
可以通过集群中的任何一个节点的JMX控制台查看当前集群信息(比如:http://hostname:8080/jmx-console),然后点击 jboss:service=DefaultPartition
注意
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地址,负载平衡算法,以及当某个目标节点不可用时如何处理请求。每个请求中,服务器节点将用集群的最近的改变信息更新存根拦截器。比如,一个节点从集群中移除,当每个存根拦截器连接到任何一个激活的节点时,都将更新最新的配置信息。所有基于存根的操作对于客户来讲都是透明的,客户端拦截器集群架构如下图所示:
1.2.2 负载平衡器
别的JBoss服务,特别是HTTP WEB服务,不要求客户端下载任何东西。客户直接通过某个通信协议发送请求并接受响应,比如HTTP协议。这种情况下,一个负载平衡器用来处理所有的请求并将这些请求派发给集群中的节点。负载平衡器通常是集群的一部分,它知道集群配置和恢复策略,客户端只需要知道负载平衡器。负载平衡器架构如下图所示:
负载平衡器架构的一个潜在的问题是负载平衡器是所有请求的唯一入口,必须保证高的可用性。
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/
注意
当前存在一个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. <!--sp-->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>
armMemberService
- PartitionName 指定集群名字,缺省值为 DefaultPartition.
- URLs 指定布署器监控布署文件的目录,如果不存在,MBean将创建。 "." 表示配置目录(比如 $JBOSS_HOME/server/all/)。
- ScanPeriod
Farming 服务是URLDeploymentScanner的扩展,用来扫描deploy目录,因此在FarmMemberService MBean中可以使用URLDeploymentScanner
1.2.4 分布式状态复制服务
集群环境中,分布式状态管理是集群必须提供的一个关键服务。比如一个有状态的会话Bean应用程序,会话状态必须在所有节点的Bean实例中保持同步,因此不管哪个节点服务请求,客户端应该得到同样的会话状态。实体Bean应用程序中,Bean实例必须在集群中进行缓存以减轻数据库负载。目前JBoss AS中的状态复制和分布式缓存由两种方式提供: HASessionState
- HASessionState MBean 基于EJB2.X有状态会话Bean和JBoss3.x和4.x的HTTP负载平衡器提供会话复制和分布式缓存服务。MBean相关定义在 all/deploy/cluster-service.xml
- JBoss Cache 是一个分布式缓存框架,可以用在任何应用程序服务器环境,也可以独立使用,已经逐渐替换 HASessionState
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
27. # It will be checked for updates each 60
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
34. # later versions of mod_jk, and is needed for
35. # for
36. JkShmFile logs/jk.shm
37.
38. # Add jkstatus for
39.
40. JkMount status
41. Order deny,allow
42. Deny from all
43. 127.0.0.1
44.
其中有两个设置是非常重要的:
- LoadModule
- 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
1.5.3. 在mod_jk中配置工作者节点
接着需要创建conf/workers.properties文件,这个文件用来指定名个不同的Servlet容器的位置以及调用如何在这些容器之间进行平衡。针对一个两个节点的设置,示例文件如下:
java 代码
1. # Define list of workers that will be used
2. # for
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
28. worker.status.type=status
上述文件配置mod_jk执行Round-Robin的基于sticky-sessions的负载平衡策略,两个节点都监听8009端口。
works.properties文件中,每个节点使用worker.XXX 命名规范进行定义,这里 XXX
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
注意
A non-loadbalanced setup with a single node required the worker.list=node1
1.5.4. 配置JBoss
最后必须配置集群中所有节点的JBoss Tomcat实例。
在每个节点中,我们必须根据workers.properties文件中的worker名称来命名节点。比如,在JBoss 实例node1中,需要编辑JBOSS_HOME/server/all/deploy/jbossweb-tomcat50.sar/server.xml 文件,定位
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
xml 代码
1. <attribute name="UseJK">trueattribute>
到此为止,我们已经成功设置Apache_mod_jk使用sticky-session方式的负载平衡。
注意
更多信息可以参考http://wiki.jboss.org/wiki/Wiki.jsp?page=UsingMod_jk1.2WithJBoss。
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_SYNC 和REPL_ASYNC,确定改变是应该同步还是异步复制。使用同步复制,确保在请求完成之前传播改变,同步复制相对来说会慢一些。
- ClusterName
- UseMarshalling 和InactiveOnStartup
- ClusterConfig 配置底层的JGoups 堆栈。最重要的配置元素是广播地址和端口号mcast_addr 和mcast_port,详细配置请参见JGroups配置。
- LockAcquisitionTimeout
- UseReplQueue 使用异步复制时是否启动复制队列,这允许多个更新一起执行从而提升性能。复制队列属性由 ReplQueueInterval 和ReplQueueMaxElements
- ReplQueueInterval
- ReplQueueMaxElements: 配置复制队列可以保存的最多的元素数目。
1.5.6. 在应用程序中启用会话复制
要在你的WEB应用程序中启用集群,必须在web.xml文件中声明distributable。示例如下:
xml 代码
1. <!--sp-->xml version="1.0"?>
2. <web-app xmlns="http://java.sun.com/xml/ns/j2ee"
3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4. xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
5. http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"
6. version="2.4">
7. <distributable/>
8. <!-- ... -->
9. web-app>
<!--l version="1.0-->
可以在jboss-web.xml文件中进一步配置会话复制,示例如下:
xml 代码
<jboss-web>
<replication-config>
<replication-trigger>SET_AND_NON_PRIMITIVE_GETreplication-trigger>
<replication-granularity>SESSIONreplication-granularity>
<replication-field-batch-mode>truereplication-field-batch-mode>
replication-config>
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
如果你的会话通常比较小,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
/* * * @@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
$ 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()