EJB 事务

IBM®
市场

搜索
提交我的 IBM
站点导航
developerWorks®
学习 开发 社区
学习Java technology
内容
概览
前言
前期准备工作
传播属性实例祥解
总结
相关主题
评论
实例详解 EJB 中的六大事务传播属性
王 汉敏
2012 年 6 月 04 日发布
WeiboGoogle+用电子邮件发送本页面
Comments 2
前言
事务 (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 中这六种不同的事务传播属性。
前期准备工作
首先,我们创建如下几个类,来作为我们后续中的实例。
清单 1. Address 实体 Bean
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@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 中添加一些字段:
清单 2. Person 实体 Bean
1
2
3
4
5
6
7
8
9
10
11
12
13
14
@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
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
@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
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@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
当一个方法的事务传播属性被设置成为“Required”的时候,说明该方法需要在事务的环境中运行。如果调用该方法的客户端不在事务中,这个时候,当该方法执行的时候就会开启一个新的事务;相反,如果调用该方法的方法已经运行在一个事务之中,那么该方法就会加入原来的事务运行。
下面举例说明一下
清单 5. 调用 ClientEJB 的 Main 函数
1
2
3
4
5
6
7
8
9
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
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
@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
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@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 注解定义
1
2
@Target(value={METHOD,TYPE})
@Retention(value=RUNTIME)
从它的定义中可以看出,这个注解可以定义在类的层次上,也可以定义在方法上。如果两个层次同时定义的话,定义在方法上的属性会覆盖定义在类层次上的属性。EJB 中类层次上的和方法层次上的传播属性默认均为”Required”。
这里我们显示地将 CommonEJB 中的 CreateAddress() 方法的传播属性设置成了”Required”,虽然我们没有必要这么做,他确省就是”Required”。我们在 ClientEJB 中的 CreatePerson() 方法中调用了这个 CreateAddress() 方法,根据”Required”传播属性的定义,CreateAddress() 方法将会加入调用者 CreatePerson() 开启的事务当中,成为其中的一部分。下面是这个程序运行的结果
图 1.GlassFish 控制台输出
图 1.GlassFish 控制台输出
图 1 显示两个方法均执正常行完毕,没有任何异常抛出。
图 2. 实体 Bean Person 对应的数据库表
图 2. 实体 Bean Person 对应的数据库表
CreatePerson() 方法正常执行完毕后,ID 为 200 的人被持久化到数据库中。
图 3. 实体 Bean Address 对应的数据库表
图 3. 实体 Bean Address 对应的数据库表
CreateAddress() 方法正常执行完毕后,ID 为 100 的地址被持久化到数据库中。
这就说明这两个方法均在事务的环境中进行了持久化的操作,且没有回滚。
下面我们使用依赖注入,将 SessionContext 注入到 ClientEJB,并在 createPerson() 中调用 setRollbackOnly() 将这个方法所在的事务设置成 Doomed 事务。Doomed 事务就是那些铁定要回滚的事务,无论他进行了什么操作,无论成功与否,都要回滚,这就是他的宿命。
清单 9. 客户端 createPerson() 回滚
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
@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() 回滚
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
@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
当一个方法的事务传播属性被设置成为“Requires_New”的时候。如果调用该方法的客户端不在事务中,这个时候,当该方法执行的时候就会开启一个新的事务;相反,如果调用该方法的方法已经运行在一个事务之中,那么该方法同样会开启一个新的事务,并在新事物中运行,而原来的事务则挂起,等待新开启的事情执行完毕之后再继续执行。
清单 11 .Requires_New 属性中 ClientEJB 客户端
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@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
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@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!--------"); 

}
}
当我们运行程序的时候其结果如下,说明程序正确执行完毕,数据也持久化到了数据库中
图 4 .Person 表
图 4 .Person 表
CreatePerson() 方法正常执行完毕后,ID 为 88 的人被持久化到数据库中。
图 5 .Address 表
图 5 .Address 表
CreateAddress() 方法正常执行完毕后,ID 为 55 的地址被持久化到数据库中。
下面我们将 ClientEJB 设置为 Doomed 事务,而保持 CommonEJB 不变,看看是什么情况。
外围事务回滚,不影响新开启的事务。
清单 13. Requires_New 属性中被设定为 Doomed 的 ClientEJB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
@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 被持久化到了数据库中。
图 6.Address 表
图 6.Address 表
这就因为 createPerson() 所在的事务进行了回滚,而 createAddress() 所在的事务没有回滚。
内围事务回滚,不影响外围事务。
下面我们将 CommonEJB 设置为 Doomed 事务,而保持 ClientEJB 不变,看看是什么情况。
清单 14.Requires_New 属性中被设定为 Doomed 的 CommonEJB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
@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() 没有。
图 7.Person 表
图 7.Person 表
以上两种情况说明,Require_New 开启的是一个新事务,外围事务也就是调用者或客户端所在事务的回滚,不会影响到这个新开起的事务;同样,新开起的事务的回滚与否对外围事务也没有任何影响。
Supports
当一个方法的事务传播属性被设置成为“Supports”的时候,如果调用该方法的客户端不在事务中,这个时候,当该方法执行的时候不会开启一个新的事务,仍会在无事务的环境中运行;相反,如果调用该方法的方法已经运行在一个事务之中,那么该方法就会加入原来的事务运行,这点和“Required”相同。
清单 15.Supports 属性中 ClientEJB 客户端
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
@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“。
清单 16.Supports 属性中 CommonEJB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@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!--------"); 

}
}
当我们执行之后,可以看到两个方法中的数据均持久化到了数据库中
图 8.Person 表
图 8.Person 表
图 9.Address 表
图 9.Address 表
当调用者自己新开起一个事务,或已经处于某个事务之中的时候,被调用者会加入调用者的事务。这样调用者和被调用者就处于同一个事务之中,在任何一个方法内出现引起事务回滚的事件,调用者和被调用者都要回滚。
当调用者不在事务中运行,而被调用者的事务传播属性为“SUPPORTS”时,被调用者也不会开启新事务,仍旧在无事务的环境中运行,这个时候和普通得 Java 方法之间的调用毫无区别,任何一个 Java 程序员对这种情况都会司空见惯,这里也不再赘叙。
Not_Supported
当一个方法的事务传播属性被设置成为“Not_Supported”的时候,如果调用该方法的客户端不在事务中,这个时候,当该方法执行的时候也不会开启一个新的事务,而仍是在无事务中的环境中运行,这点和用无事务的方法调用传播属性为”Suppots”的方法相同,不再赘叙;相反,如果调用该方法的客户端已经运行在一个事务之中,那么原来的事务则挂起,等待被调用者在无事务的环境中运行完毕之后,再继续执行原来挂起的事务,原来事务的回滚与否对被调用者均无影响。
清单 17.Not_Supported 属性中 ClientEJB 客户端
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
@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
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
@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() 执行完毕,继续事务的过程中完成了提交。
图 10.Person 表
图 10.Person 表
如下所示,我们将 createPerson() 方法新开起的事务设定成 Doomed 事务,执行完毕后,由于事务的回滚,Person 数据并没有持久化到数据库中。而 createAddress() 方法一直在无事务的环境中运行,所以当外围事务回滚的时候,对他并没有人很影响。
清单 19.Not_Supported 属性中 ClientEJB 客户端
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
@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
当一个方法的事务传播属性被设置成为“Mandatory”的时候,说明这个方法必须在一个事务环境下执行。如果调用该方法的客户端不在事务中,这个时候,调用该方法时,就会抛出 javax.ejb.EJBTransactionRequiredException 异常;相反,如果调用该方法的方法已经运行在一个事务之中,那么该方法就会加入原来的事务运行,这点和“Required”相同。
清单 20.Mandatory 属性中 ClientEJB 客户端
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@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”。
清单 21.Mandatory 属性中 CommonEJB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@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 事务后,事务中的所有方法的持久化操作都会随着事务的回滚而回滚。这里不再重复举例。
图 11.Person 表
图 11.Person 表
图 12.Address 表
图 12.Address 表
如清单 22 我们把 ClientEJB 的传播属性修改为”Mandatory”,因为 Main() 不在事务中运行,所以在 Main() 方法中调用 ClientEJB 是就会抛出异常。
例如:
清单 22.Mandatory 属性中 ClientEJB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
@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
图 13.EJBTransactionRequiredException
Never
当一个方法的事务传播属性被设置成为“Never”的时候,说明这个方法必须在无事务环境下执行。如果调用该方法的方法不在事务中,该方法将会在无事务环境中运行;相反,如果调用该方法的方法已经运行在一个事务之中,那么调用该方法的时候就会抛出“RemoteException”异常。
清单 23.Never 属性中的 ClientEJB 客户端
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@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”的传播属性。
清单 24.Never 属性中的 CommonEJB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@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!--------"); 

}
}
当我们运行的时候控制台抛出了如下图所示的异常,由于程序未能正确执行,所以方法内持久化操作均不能完成,数据库中也没有相应的数据。
图 14.Never 中抛出异常
图 14.Never 中抛出异常
综上所述,EJB 中事务的传播属性可以用如下表格进行概括:
表 1. 事务传播属性总表
事务传播属性 客户端在事务中运行
MANDATORY 加入客户端所在的事务
NEVER 抛出 RemoteException 异常
NOT_SUPPORTED 客户端的事务挂起,方法在无事务环境中运行,客户端的事务继续执行
REQUIRED 加入客户端所在的事务
REQUIRES_NEW 客户端的事务挂起,为该方法新开起一个事务,客户端的事务继续执行
SUPPORTS 加入客户端所在的事务
Nested
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 编程各个方面的文章。
评论
添加或订阅评论,请先登录或注册。
有新评论时提醒我
最新的历史热门
2014 年 05 月 06 日
Java-dev
不错,思路清晰,言简意赅,做读书笔记正好!
2014 年 05 月 06 日
Java-dev
不错,思路清晰,言简意赅,做读书笔记正好!
回页首
developerWorks
站点反馈
我要投稿
投稿指南
报告滥用
第三方提示
关注微博
加入
ISV 资源 (英语)
选择语言
English
中文
日本語
Русский
Português (Brasil)
Español
한글
技术文档库
订阅源
社区
dW 中国时事通讯
软件下载
联系 IBM 隐私条约 使用条款 信息无障碍选项 反馈 Cookie 首选项
User Research Participants Wanted
Sign me up!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值