事务 (Transaction) 是访问并可能更新数据库中各种数据项的一个程序执行单元 (unit)。在关系数据库中,一个事务可以是一条或一组 SQL 语句,甚至整个程序。它有通常被称为 ACID 的原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持续性(Durability)四大特性:
原子性(Atomicity):一个事务是一个不可分割的工作单位,事务中包括的诸操作要么都做,要么都不做。
一致性(Consistency):事务必须是使数据库从一个一致性状态变到另一个一致性状态。一致性与原子性是密切相关的。
隔离性(Isolation):一个事务的执行不能被其他事务干扰。即一个事务内部的操作及使用的数据对并发的其他事务是隔离的,并发执行的各个事务之间不能互相干扰。
持久性(Durability):持续性也称永久性(permanence),指一个事务一旦提交,它对数据库中数据的改变就应该是永久性的。接下来的其他操作或故障不应该对其有任何影响。
在 Java EE 的应用开发中,事务的应用是必不可少的,同时由于方法调用的原因,比如方法 A 调用方法 B 的时候。如果方法 A 在事务环境中运行,那么方法 B 是否也要在事务中运行呢,方法 B 是要和方法 A 在同一个事务中运行还是新开起一个事物呢?等等。要弄清楚这些问题,就要牵涉到事务传播属性的问题,EJB 中针对不同的的情况提供了下面六种不同的事物传播属性:
Required:用该属性标注的方法或组件总是在事务中运行。如果客户端已经在事务中,则在原事务中运行;如果没有事务,则开启一个新事务,在其中运行。
Requires_New:方法或组件总是在新开启的事务中运行。如果客户端已经在事务中,则首先将原事务挂起,然后新开启一个事务,在其中运行,新事务结束之后,原来事务从挂起点继续执行;如果没有事务,则开启一个新事务,在其中运行。
Supports:和 Required 属性的不同点是,在没有事务的环境中不会开启一个新事务;如果存在事务的话则加入其中运行,这点和 Reuqired 相同。
Not_Supported:如果事务已经存在的情况下,则原来的事务要挂起,然后调用标注该属性的方法或组件,调用结束之后,继续原来的事务;无事务环境中调用的时候,不开启新事务,这点和 Supports 相同。
Mandatory:调用标注该属性的方法或组件的客户端,必须已经在事务中,如果不在事务中则会抛出异常;如果已经在事务中,则加入原来事务运行。和 Required 不同的是,该属性不会自动开启新的事务;
Never:用 Never 属性标注的方法或组件,不能在事务中运行。如果调用该方法或组件的客户端已经在事务中,则抛出异常。
下面就实例详细介绍一下 EJB 中这六种不同的事务传播属性。
首先,我们创建如下几个类,来作为我们后续中的实例。
@Entity public class Address implements Serializable { private static final long serialVersionUID = 1L; @Id @GeneratedValue(strategy = GenerationType.AUTO) private Long id; private String country; private String city; private String street; private String postCode; private String TsAttribute; @Temporal(javax.persistence.TemporalType.TIMESTAMP) private Date createTime; // Getters and Setters } |
我们在 Address 实体 Bean 中添加一些字段:
@Entity public class Person implements Serializable { private static final long serialVersionUID = 1L; @Id @GeneratedValue(strategy = GenerationType.AUTO) private Long id; private String firstName; private String lastName; private int age; private String TsAttribute; @Temporal(javax.persistence.TemporalType.TIMESTAMP) private Date createTime; // Getters and Setters } |
同样我们在 Person 实体 Bean 中添加一些字段:
清单 3. 无状态的 SessionBean CommonEJB
@Stateless public class CommonEJB implements CommonEJBRemote { @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override public void createAddress() { Address address = new Address(); /* * * 对 address 对象 * 的属性进行赋值 * */ em.persist(address); } } |
我们在 CommonEJB 中创建了一个名为 createAddress() 的业务方法,使用这个方法来持久化 Address 实体 bean,因此我们也使用 @PersistenceContext 注入了相应的持久化单元,我们将会将 Address 持久化到这个持久化单元对应的数据库中。
清单 4. 无状态的 SessionBean ClientEJB
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override public void createPerson() { Person person = new Person(); /* * * 对 person 对象 * 的属性进行赋值 * */ em.persist(person); commonEJB.createAddress();// 调用 CommonEJB 中的 createAddress 方法 } } |
同样,我们在 ClientEJB 创建了一个名为 createPerson() 的业务方法,使用这个方法来持久化 Person 实体 bean。稍有不同的是我们不仅注入了相应的持久化单元,而且注入了 CommonEJB 到这个 EJB 中,并且在 createPerson() 方法中调用了 CommonEJB 中的业务方法。
好了现在,我们所有的准备工作都已完成下面我们开始逐一介绍这六大事务传播属性。
当一个方法的事务传播属性被设置成为“Required”的时候,说明该方法需要在事务的环境中运行。如果调用该方法的客户端不在事务中,这个时候,当该方法执行的时候就会开启一个新的事务;相反,如果调用该方法的方法已经运行在一个事务之中,那么该方法就会加入原来的事务运行。
下面举例说明一下
public class Main { @EJB private static ClientEJBRemote clientEJB; public static void main(String[] args) { clientEJB.createPerson(); } } |
我们使用 NetBeans 中独有的技术来调用 EJB,这样的话我们就不需要手动使用 JNDI 了,直接使用依赖注入的技术,在 main() 方法中注入我们要调用的 EJB,这里我们调用的是名为 ClientEJB 的 EJB。根据自 Java EE 5 引入的“Configuration by Exception”的原则,确省情况下这个 EJB 使用的是”Required”这个事务传播属性,根据”Required”事务传播属性的要求,ClientEJB 被 main() 方法调用的时候会开启一个新的事务。
清单 6.ClientEJB 持久化 Person 实体 Bean
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override public void createPerson() { Person person = new Person(); person.setId(100l); person.setFirstName("Leo"); person.setLastName("Wang"); person.setAge(88); person.setTsAttribute("Required"); person.setCreateTime(new Date()); em.persist(person); commonEJB.createAddress(); System.out.println("----ClientEJB Excute completely!-------"); } } |
在 ClientEJB 这个无状态 EJB 中的 createPerson() 业务方法中我们持久化了一个 Person 对象到数据库中。
清单 7.CommonEJB 持久化 Address 实体 Bean
@Stateless public class CommonEJB implements CommonEJBRemote { @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override @TransactionAttribute(TransactionAttributeType.REQUIRED) public void createAddress() { Address address = new Address(); address.setId(200l); address.setCountry("China"); address.setCity("Beijing"); address.setStreet("Long Jin"); address.setPostCode("102208"); address.setTsAttribute("REQUIRED"); address.setCreateTime(new Date()); em.persist(address); System.out.println("------ClientEJB Excute completely!---------"); } } |
在 CommonEJB 这个无状态 EJB 中的 createAddress () 业务方法中我们持久化了一个 Person 对象到数据库中。
TransactionAttribute 这个注解的定义如下:
清单 8.TransactionAttribute 注解定义
@Target(value={METHOD,TYPE}) @Retention(value=RUNTIME) |
从它的定义中可以看出,这个注解可以定义在类的层次上,也可以定义在方法上。如果两个层次同时定义的话,定义在方法上的属性会覆盖定义在类层次上的属性。EJB 中类层次上的和方法层次上的传播属性默认均为”Required”。
这里我们显示地将 CommonEJB 中的 CreateAddress() 方法的传播属性设置成了”Required”,虽然我们没有必要这么做,他确省就是”Required”。我们在 ClientEJB 中的 CreatePerson() 方法中调用了这个 CreateAddress() 方法,根据”Required”传播属性的定义,CreateAddress() 方法将会加入调用者 CreatePerson() 开启的事务当中,成为其中的一部分。下面是这个程序运行的结果
图 1 显示两个方法均执正常行完毕,没有任何异常抛出。
CreatePerson() 方法正常执行完毕后,ID 为 200 的人被持久化到数据库中。
CreateAddress() 方法正常执行完毕后,ID 为 100 的地址被持久化到数据库中。
这就说明这两个方法均在事务的环境中进行了持久化的操作,且没有回滚。
下面我们使用依赖注入,将 SessionContext 注入到 ClientEJB,并在 createPerson() 中调用 setRollbackOnly() 将这个方法所在的事务设置成 Doomed 事务。Doomed 事务就是那些铁定要回滚的事务,无论他进行了什么操作,无论成功与否,都要回滚,这就是他的宿命。
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Resource private SessionContext ctx; @Override public void createPerson() { Person person = new Person(); person.setId(100l); person.setFirstName("Leo"); person.setLastName("Wang"); person.setAge(88); person.setHeight(170); person.setWeight(65); person.setCreateTime(new Date()); em.persist(person); commonEJB.createAddress(); ctx.setRollbackOnly(); System.out.println("-----ClientEJB Excute completely!---------"); } } |
这个时候 我们保持 CommonEJB 不变,然后执行这个程序。
使用同样的办法,我们将 ClientEJB 保持不变,将 CommonEJB 中的 createAddress() 方法所在的事务设置成 Doomed 事务,然后执行程序。
清单 10. 被调用者 createAddress() 回滚
@Stateless public class CommonEJB implements CommonEJBRemote { @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Resource private SessionContext ctx; @Override @TransactionAttribute(TransactionAttributeType.REQUIRED) public void createAddress() { Address address = new Address(); address.setId(200l); address.setCountry("China"); address.setCity("Beijing"); address.setStreet("Long Jin"); address.setPostCode("102208"); address.setCreateTime(new Date()); em.persist(address); ctx.setRollbackOnly(); System.out.println("--------ClientEJB Excute completely!--------"); } } |
上述两种情况,从控制台上我们均可以看出两个方法正确执行完毕,但是两个方法中数据均没有被持久化到数据库中,这就是说,他们所在的事务一起进行了回滚。出现这样的情况是因为,”Required”属性是加入原有事务,也就是说它们处于同一个事物当中,一方滚另一方也回滚。
当一个方法的事务传播属性被设置成为“Requires_New”的时候。如果调用该方法的客户端不在事务中,这个时候,当该方法执行的时候就会开启一个新的事务;相反,如果调用该方法的方法已经运行在一个事务之中,那么该方法同样会开启一个新的事务,并在新事物中运行,而原来的事务则挂起,等待新开启的事情执行完毕之后再继续执行。
清单 11 .Requires_New 属性中 ClientEJB 客户端
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override public void createPerson() { Person person = new Person(); person.setId(88l); person.setFirstName("Tom"); person.setLastName("Zhang"); person.setAge(88); person.setTsAttribute("Required"); person.setCreateTime(new Date()); em.persist(person); commonEJB.createAddress(); System.out.println("------ClientEJB Excute completely!---------"); } } |
我们仍然保持 ClientEJB 的默认属性不变,而仅仅将 CommonEJB 中的 createAddress() 方法的事务传播属性设置成”Requires_New”,如清单 12 所示
清单 12. Requires_New 属性中 CommonEJB
@Stateless public class CommonEJB implements CommonEJBRemote { @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override @TransactionAttribute(TransactionAttributeType.REQUIRES_NEW) public void createAddress() { Address address = new Address(); address.setId(55l); address.setCountry("China"); address.setCity("Shanghai"); address.setStreet("Long Jin"); address.setPostCode("102208"); address.setTsAttribute("REQUIRES_NEW"); address.setCreateTime(new Date()); em.persist(address); System.out.println("----------ClientEJB Excute completely!--------"); } } |
当我们运行程序的时候其结果如下,说明程序正确执行完毕,数据也持久化到了数据库中
CreatePerson() 方法正常执行完毕后,ID 为 88 的人被持久化到数据库中。
CreateAddress() 方法正常执行完毕后,ID 为 55 的地址被持久化到数据库中。
下面我们将 ClientEJB 设置为 Doomed 事务,而保持 CommonEJB 不变,看看是什么情况。
外围事务回滚,不影响新开启的事务。
清单 13. Requires_New 属性中被设定为 Doomed 的 ClientEJB
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Resource SessionContext ctx; @Override public void createPerson() { Person person = new Person(); person.setId(88l); person.setFirstName("Tom"); person.setLastName("Zhang"); person.setAge(88); person.setTsAttribute("Required"); person.setCreateTime(new Date()); em.persist(person); ctx.setRollbackOnly(); commonEJB.createAddress(); System.out.println("------ClientEJB Excute completely!---------"); } } |
当我们执行完程序之后,Glass Fish 控制台显示程序正确执行没有异常抛出,但是数据库中显示只有 Address 被持久化到了数据库中。
这就因为 createPerson() 所在的事务进行了回滚,而 createAddress() 所在的事务没有回滚。
内围事务回滚,不影响外围事务。
下面我们将 CommonEJB 设置为 Doomed 事务,而保持 ClientEJB 不变,看看是什么情况。
清单 14.Requires_New 属性中被设定为 Doomed 的 CommonEJB
@Stateless public class CommonEJB implements CommonEJBRemote { @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Resource SessionContext ctx; @Override @TransactionAttribute(TransactionAttributeType.REQUIRES_NEW) public void createAddress() { Address address = new Address(); address.setId(55l); address.setCountry("China"); address.setCity("Shanghai"); address.setStreet("Long Jin"); address.setPostCode("102208"); address.setTsAttribute("REQUIRES_NEW"); address.setCreateTime(new Date()); em.persist(address); ctx.setRollbackOnly(); System.out.println("----------ClientEJB Excute completely!--------"); } } |
当我们正确执行完程序之后,数据库中只有 Person 的记录,这就说明 createAddress() 所在的方法进行了回滚,而 createPerson() 没有。
以上两种情况说明,Require_New 开启的是一个新事务,外围事务也就是调用者或客户端所在事务的回滚,不会影响到这个新开起的事务;同样,新开起的事务的回滚与否对外围事务也没有任何影响。
当一个方法的事务传播属性被设置成为“Supports”的时候,如果调用该方法的客户端不在事务中,这个时候,当该方法执行的时候不会开启一个新的事务,仍会在无事务的环境中运行;相反,如果调用该方法的方法已经运行在一个事务之中,那么该方法就会加入原来的事务运行,这点和“Required”相同。
清单 15.Supports 属性中 ClientEJB 客户端
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override public void createPerson() { Person person = new Person(); person.setId(33l); person.setFirstName("Jerry"); person.setLastName("Leoo"); person.setAge(22); person.setTsAttribute("Required"); person.setCreateTime(new Date()); em.persist(person); commonEJB.createAddress(); System.out.println("------ClientEJB Excute completely!---------"); } } |
我们仍然保持 ClientEJB 的默认属性不变,而仅仅将 CommonEJB 中的 createAddress() 方法的事务属性设置成“Supports“。
@Stateless public class CommonEJB implements CommonEJBRemote { @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override @TransactionAttribute(TransactionAttributeType.SUPPORTS) public void createAddress() { Address address = new Address(); address.setId(66l); address.setCountry("USA"); address.setCity("NewYork"); address.setStreet("Seventh Avenue"); address.setPostCode("123-456"); address.setTsAttribute("SUPPORTS"); address.setCreateTime(new Date()); em.persist(address); System.out.println("----------ClientEJB Excute completely!--------"); } } |
当我们执行之后,可以看到两个方法中的数据均持久化到了数据库中
当调用者自己新开起一个事务,或已经处于某个事务之中的时候,被调用者会加入调用者的事务。这样调用者和被调用者就处于同一个事务之中,在任何一个方法内出现引起事务回滚的事件,调用者和被调用者都要回滚。
当调用者不在事务中运行,而被调用者的事务传播属性为“SUPPORTS”时,被调用者也不会开启新事务,仍旧在无事务的环境中运行,这个时候和普通得 Java 方法之间的调用毫无区别,任何一个 Java 程序员对这种情况都会司空见惯,这里也不再赘叙。
当一个方法的事务传播属性被设置成为“Not_Supported”的时候,如果调用该方法的客户端不在事务中,这个时候,当该方法执行的时候也不会开启一个新的事务,而仍是在无事务中的环境中运行,这点和用无事务的方法调用传播属性为”Suppots”的方法相同,不再赘叙;相反,如果调用该方法的客户端已经运行在一个事务之中,那么原来的事务则挂起,等待被调用者在无事务的环境中运行完毕之后,再继续执行原来挂起的事务,原来事务的回滚与否对被调用者均无影响。
清单 17.Not_Supported 属性中 ClientEJB 客户端
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override public void createPerson() { Person person = new Person(); person.setId(123l); person.setFirstName("Marry"); person.setLastName("Bush"); person.setAge(22); person.setTsAttribute("Required"); person.setCreateTime(new Date()); em.persist(person); commonEJB.createAddress(); System.out.println("------ClientEJB Excute completely!---------"); } } |
ClientEJB 中 createPerson() 方法我们仍然使用默认传播属性,而将 CommonEJB 中的 createAddress() 的传播属性设置成了 Not_Supported。
清单 18.Not_Supported 属性中 CommonEJB
@Stateless public class CommonEJB implements CommonEJBRemote { @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override @TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED) public void createAddress() { /* * 此处省略了一些业务处理的代码 , * 因为 persist(Object obj) 必须在事务中执行 * 所以此处暂时不能使用 * */ System.out.println("----------ClientEJB Excute completely!--------"); } } |
运行程序完毕之后,从图 10 中可以看到 ID 为 123 的 Person 已经持久化到了数据库中,说明 createPerson() 所在事务在调用 createAddress() 方法时,把事务挂起,当 createAddress() 执行完毕,继续事务的过程中完成了提交。
如下所示,我们将 createPerson() 方法新开起的事务设定成 Doomed 事务,执行完毕后,由于事务的回滚,Person 数据并没有持久化到数据库中。而 createAddress() 方法一直在无事务的环境中运行,所以当外围事务回滚的时候,对他并没有人很影响。
清单 19.Not_Supported 属性中 ClientEJB 客户端
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Resource SessionContext ctx; @Override public void createPerson() { Person person = new Person(); person.setId(123l); person.setFirstName("Marry"); person.setLastName("Bush"); person.setAge(22); person.setTsAttribute("Required"); person.setCreateTime(new Date()); em.persist(person); ctx.setRollbackOnly(); commonEJB.createAddress(); System.out.println("------ClientEJB Excute completely!---------"); } } |
当一个方法的事务传播属性被设置成为“Mandatory”的时候,说明这个方法必须在一个事务环境下执行。如果调用该方法的客户端不在事务中,这个时候,调用该方法时,就会抛出 javax.ejb.EJBTransactionRequiredException 异常;相反,如果调用该方法的方法已经运行在一个事务之中,那么该方法就会加入原来的事务运行,这点和“Required”相同。
清单 20.Mandatory 属性中 ClientEJB 客户端
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override public void createPerson() { Person person = new Person(); person.setId(88l); person.setFirstName("Min"); person.setLastName("Zhao"); person.setAge(22); person.setTsAttribute("Required"); person.setCreateTime(new Date()); em.persist(person); commonEJB.createAddress(); System.out.println("------ClientEJB Excute completely!---------"); } } |
我们仍然保持了 ClientEJB 中 createPerson() 方法的默认传播属性,而将 CommonEJB 中的 createAddress() 方法事务传播属性设置成了”Mandatory”。
@Stateless public class CommonEJB implements CommonEJBRemote { @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override @TransactionAttribute(TransactionAttributeType.MANDATORY) public void createAddress() { Address address = new Address(); address.setId(66l); address.setCountry("Japan"); address.setCity("Tokyo"); address.setStreet("Seventh Avenue"); address.setPostCode("444-789"); address.setTsAttribute("MANDATORY"); address.setCreateTime(new Date()); em.persist(address); System.out.println("----------ClientEJB Excute completely!--------"); } } |
运行结果如下图所示,说明两个方法都是在事务中得到了执行。由于使用“Mandatory”传播属性的方法是加入原来的外围事务,也就是说它们处于同一个事务当中,所以在任何一个方法中如果调用 setRollbackOnly() 方法将事务设定成 Doomed 事务后,事务中的所有方法的持久化操作都会随着事务的回滚而回滚。这里不再重复举例。
如清单 22 我们把 ClientEJB 的传播属性修改为”Mandatory”,因为 Main() 不在事务中运行,所以在 Main() 方法中调用 ClientEJB 是就会抛出异常。
例如:
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override @TransactionAttribute(TransactionAttributeType.MANDATORY) public void createPerson() { Person person = new Person(); person.setId(88l); person.setFirstName("Min"); person.setLastName("Zhao"); person.setAge(22); person.setTsAttribute("Required"); person.setCreateTime(new Date()); em.persist(person); commonEJB.createAddress(); System.out.println("------ClientEJB Excute completely!---------"); } } |
当我们再次运行的时候,控制台就会抛出如下异常:
图 13.EJBTransactionRequiredException
当一个方法的事务传播属性被设置成为“Never”的时候,说明这个方法必须在无事务环境下执行。如果调用该方法的方法不在事务中,该方法将会在无事务环境中运行;相反,如果调用该方法的方法已经运行在一个事务之中,那么调用该方法的时候就会抛出“RemoteException
”异常。
清单 23.Never 属性中的 ClientEJB 客户端
@Stateless public class ClientEJB implements ClientEJBRemote { @EJB private CommonEJBRemote commonEJB; @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override public void createPerson() { Person person = new Person(); person.setId(88l); person.setFirstName("Ying"); person.setLastName("Tong"); person.setAge(22); person.setTsAttribute("Required"); person.setCreateTime(new Date()); em.persist(person); commonEJB.createAddress(); System.out.println("------ClientEJB Excute completely!---------"); } } |
ClientEJB 中 createPerson() 方法采用了默认的传播属性,CommonEJB 中 createAddress() 方法使用了”Never”的传播属性。
@Stateless public class CommonEJB implements CommonEJBRemote { @PersistenceContext(unitName = "Transaction-ejbPU") EntityManager em; @Override @TransactionAttribute(TransactionAttributeType.NEVER) public void createAddress() { Address address = new Address(); address.setId(66l); address.setCountry("Korea"); address.setCity("Souel"); address.setStreet("Tian Jian"); address.setPostCode("4444444"); address.setTsAttribute("NEVER"); address.setCreateTime(new Date()); em.persist(address); System.out.println("----------ClientEJB Excute completely!--------"); } } |
当我们运行的时候控制台抛出了如下图所示的异常,由于程序未能正确执行,所以方法内持久化操作均不能完成,数据库中也没有相应的数据。
综上所述,EJB 中事务的传播属性可以用如下表格进行概括:
MANDATORY |
加入客户端所在的事务 | 抛出 TransactionRequiredException 异常 |
NEVER |
抛出 RemoteException 异常 |
方法在无事务环境中运行 |
NOT_SUPPORTED |
客户端的事务挂起,方法在无事务环境中运行,客户端的事务继续执行 | 方法在无事务环境中运行 |
REQUIRED |
加入客户端所在的事务 | 为该方法新开起一个事务 |
REQUIRES_NEW |
客户端的事务挂起,为该方法新开起一个事务,客户端的事务继续执行 | 为该方法新开起一个事务 |
SUPPORTS |
加入客户端所在的事务 | 方法在无事务环境中运行 |
Spring 中还有一个没有在 EJB 标准中定义的 Nested 的事务传播属性,这个属性和 Requires_New 极为类似,同样被已在事务环境中的 Client 调用的时候须开启一个新事务。唯一不同的就是 Nested 事务传播属性,更像一个嵌套属性,也就是说它新开起的这个事物要依附于父事务,如果父事务提交或者回滚了,它也需要跟着提交或者回滚。而 Requies_New 新开起的事务和父事务没有这种关系,它们是完全独立的两个事务,一方提交或者回滚不影响另一方。因为这个并不是 Java EE 标准中的事务传播属性,所以在这里也就不再赘叙了,有兴趣的读者可以参看相关资料。
逻辑层是 Java EE 项目开发中的核心层,而事物是逻辑层中的灵魂所在,所以掌握好逻辑层中的事物处理方式,对 Java EE 开发起到关键的作用,虽然开源领域中,事务处理的方式各有不同,但是在逻辑层事物传播属性中,上面所提到的六大传播属性,已经被大家认可并广泛接受,已经被不同的逻辑层中间件普遍采用,使用已经很广,所以掌握了上面的那些传播属性的使用方法,我们就可以在以后 Java EE 项目逻辑层的开发中以不变应万变了。
学习
-
Enterprise JavaBeans 3.1:EJB 是 sun 的服务器端组件模型,设计目标与核心应用是部署分布式应用程序。凭借 java 跨平台的优势,用 EJB 技术部署的分布式系统可以不限于特定的平台。EJB (Enterprise JavaBean) 是 J2EE 的一部分,定义了一个用于开发基于组件的企业多重应用程序的标准。其特点包括网络服务支持和核心开发工具 (SDK)。在 J2EE 里,Enterprise Java Beans(EJB) 称为 Java 企业 Bean,是 Java 的核心代码,分别是会话 Bean(Session Bean),实体 Bean(Entity Bean)和消息驱动 Bean(MessageDriven Bean)。从 EJB3.1 开始原来的实体 Bean (Entity Bean)已经独立出来作为数据持久化的技术规范— -JPA 而单独存在,并且从 EJB 3.1 起,EJB 又添加了一个新成员单例会话 Bean(Singleton Session Bean)。
-
Spring In Action:Spring 是一个开源框架,它由 Rod Johnson 创建。它是为了解决企业应用开发的复杂性而创建的。Spring 使用基本的 JavaBean 来完成以前只可能由 EJB 完成的事情。然而,Spring 的用途不仅限于服务器端的开发。从简单性、可测试性和松耦合的角度而言,任何 Java 应用都可以从 Spring 中受益。
- “Spring 事务管理高级应用难点剖析,第 1 部分”(developerWorks,2010 年 3 月):Spring 的事务管理是被使用得最多的功能之一,虽然 Spring 事务管理已经帮助程序员将要做的事情减到了最小。但在实际开发中,如果使用不当,依然会造成数据连接泄漏等问题。本系列以实际应用中所碰到的各种复杂的场景为着眼点,对这些应用的难点进行深度的剖析。
- “Spring 事务管理高级应用难点剖析:第 2 部分”(developerWorks,2010 年 3 月):在本文中,作者将继续深入剖析在实际 Spring 事务管理应用中容易遇见的一些难点,包括混合使用多种数据访问技术(如 Spring JDBC + Hibernate)的事务管理问题,以及通过 Spring AOP 增强的 Bean 存在的一些比较特殊的情况。
- “Spring 事务管理高级应用难点剖析:第 3 部分”(developerWorks,2010 年 3 月):本文是“Spring 事务管理高级应用难点剖析”系列文章的第 3 部分,作者将继续深入剖析在实际 Spring 事务管理应用中容易遇见的一些难点,包括在使用 Spring JDBC 时如果直接获取 Connection,哪些情况会造成数据连接的泄漏与如何应对,以及除 Spring JDBC 外,其它数据访问技术数据连接泄漏的应对方案。
-
developerWorks Java 技术专区:这里有数百篇关于 Java 编程各个方面的文章。
讨论
- 加入 developerWorks 中文社区。查看开发人员推动的博客、论坛、组和维基,并与其他 developerWorks 用户交流。
相关推荐
在深入探讨EJB(Enterprise JavaBeans)的事务属性之前,我们先来理解一下EJB的基本概念及其在企业级应用中的重要性。EJB是Java EE平台的核心组件之一,主要用于构建可扩展、健壮且安全的企业级应用程序。它提供了一...
在EJB中,事务管理是核心功能之一,确保了数据的一致性和完整性。事务是数据库操作的基本单元,确保一组操作要么全部成功,要么全部失败,避免数据的不一致。 事务的界定是定义事务的生命周期,包括开始、挂起、...
【ejb3.0 分布式事务详解】 在企业级Java应用中,EJB(Enterprise JavaBeans)3.0提供了一种强大而灵活的方式来处理分布式事务。分布式事务是指跨越多个数据库或资源管理器的单一逻辑操作,确保了数据的一致性和...
**企业级JavaBeans(Enterprise JavaBeans,简称EJB)**是Java平台上的核心组件之一,用于构建可扩展、安全且事务处理能力强的企业级应用程序。EJB应用开发详解涵盖了这一技术的各个方面,旨在帮助开发者深入理解和...
在EJB中,JNDI用于查找和访问Bean的实例。以下是一些可能包含在`jndi.properties`中的配置项: 1. **ejb.context.props**:定义EJB上下文属性,如服务器URL、端口等。 2. **java.naming.factory.initial**:指定...
实体Bean是EJB中最基本的组件之一,用于表示持久化对象。根据领域实体分析的结果,我们可以直接将实体映射为实体Bean,并为其添加必要的属性字段。为了满足EJB规范中的findByPrimaryKey方法要求,每个实体Bean都需要...
EJB容器提供了强大的事务管理支持,它自动处理事务的开始、提交、回滚等操作,确保了业务逻辑中数据处理的原子性、一致性、隔离性和持久性(ACID属性)。开发者无需编写复杂逻辑来管理事务,这减少了错误发生的可能...
EJB 3是其第三个主要版本,发布于2006年,它引入了许多重大改进,极大地简化了EJB的开发模式,降低了学习曲线,提高了开发效率。本实例源代码集就是针对EJB 3技术的实践应用展示。 EJB 3中的关键概念包括: 1. **...
3. **EJB2.0的Helloworld实例**:在EJB2.0的Helloworld示例中,我们通常会创建一个简单的会话bean,提供一个方法,如“sayHello”。客户端通过JNDI(Java Naming and Directory Interface)查找bean并调用该方法。这...
【J2EE架构开发EJB实例详解】 J2EE(Java 2 Platform, Enterprise Edition)是一种用于构建企业级分布式应用程序的框架,它提供了一个统一的平台来开发、部署和管理多层应用程序。J2EE的核心组件之一是Enterprise ...
10. **EJB与JMS结合**:在实例94EJB3中,可能涵盖了EJB如何与Java消息服务(JMS)集成,实现异步处理和解耦。 通过这些示例学习,开发者能够深入理解EJB在实际项目中的应用,如何利用其特性提高代码的可维护性和可...
在深入学习《J2EE经典实例详解》的过程中,你会遇到各种实际项目场景,如用户认证、权限管理、数据访问、缓存策略、负载均衡、集群部署等。通过这些实例,你可以理解如何将上述技术应用于实际工作,并提升你的Java ...
《J2EE经典实例详解》是一本专注于Java企业级应用开发的实践指南,它通过一系列精选的实例,帮助开发者深入理解和熟练运用J2EE技术。J2EE(Java 2 Platform, Enterprise Edition)是Java平台的企业版,为构建分布式...
EJB 3.0中的容器自动处理事务管理,开发者只需通过设置事务属性(如`@TransactionAttribute`)来定义事务边界。这减少了事务管理的复杂性,让开发者更专注于业务逻辑。 **7. 异步处理** EJB 3.0支持异步方法调用,...
4. **容器管理的事务**:EJB3支持自动事务管理,可以通过`@TransactionAttribute`来定义事务的传播行为和隔离级别。 **JBoss AS7 (现为WildFly)** 是一个开源的应用服务器,它是EJB3规范的一个实现,提供了运行和...
在EJB中,客户端通常通过JNDI查找服务定位到Bean实例,然后调用其方法。 **EJB与事务管理** EJB容器提供强大的事务管理能力。默认情况下,无状态会话Bean在每个方法调用中都启动一个新的事务,而有状态会话Bean和...
3. **简化事务管理**:EJB 3.0简化了事务管理,开发者可以通过`@TransactionAttribute`注解定义事务的属性,如REQUIRED、REQUIRES_NEW等,无需像EJB 2.x那样处理复杂的EJB事务API。 4. **依赖注入**:EJB 3.0引入了...
- **事务管理服务**:讲解EJB3中事务管理服务的使用方法。 - **实体生命周期**:解析实体Bean的状态及其生命周期。 - **复合主键**:说明如何在EJB3中实现复合主键。 #### 七、Web服务 (Web Service) - **创建Web...