EJB事务

转载自:
http://fansofjava.iteye.com/blog/350119
事务分本地事务和分布式事务,本地事务相对简单,这里讨论一下分布事事务。
分布式环境如何管理涉及一个以上资源的事务呢?主要是通过被称为两阶段
提交(two-phase commit)来实现的。在执行一个以上的数据库操作时,如果第一个数据库操作成功,第二个数据库操作失败,要回滚第一个操作是非常困难的。所以在提交事务之前,两阶段提交协议会询问每个资源管理器的当前事务能否提交成功,如果任何一个资源管理器表示不能提交事务,那么就回滚整个事务。

EJB有两种使用事务的方式。
第一种方式通过容器管理的事务,叫CMT,另一种通过bean管理的事务叫BMT。

在CMT中,容器自动提供事务的开始、提交和回滚。总是在业务方法的开始和结束处标记事务的边界。下面以ejb3 in action上的例子说明:

Java代码 收藏代码

1. @Stateless
2. @TransactionManagement(TransactionManagementType.CONTAINER)
3. public class OrderManagerBean {
4. @Resource
5. private SessionContext context;
6. …
7. @TransactionAttribute(TransactionAttributeType.REQUIRED)
8.
9. public void placeSnagItOrder(Item item, Customer customer){
10. try {
11. if (!bidsExisting(item)){
12. validateCredit(customer);
13. chargeCustomer(customer, item);
14. removeItemFromBidding(item);
15. }
16. } catch (CreditValidationException cve) {
17. context.setRollbackOnly();
18. } catch (CreditProcessingException cpe){
19. context.setRollbackOnly();
20. } catch (DatabaseException de) {
21. context.setRollbackOnly();
22. }
23. }
24. }

@Stateless
@TransactionManagement(TransactionManagementType.CONTAINER)
public class OrderManagerBean {
@Resource
private SessionContext context;

@TransactionAttribute(TransactionAttributeType.REQUIRED)

public void placeSnagItOrder(Item item, Customer customer){
try {
if (!bidsExisting(item)){
validateCredit(customer);
chargeCustomer(customer, item);
removeItemFromBidding(item);
}
} catch (CreditValidationException cve) {
context.setRollbackOnly();
} catch (CreditProcessingException cpe){
context.setRollbackOnly();
} catch (DatabaseException de) {
context.setRollbackOnly();
}
}
}


其中,@TransactionAttribute(TransactionAttributeType.REQUIRED)表示指定事务的类型。
如果省略,默认为CMT方式。
@TransactionAttribute(TransactionAttributeType.REQUIRED)通知容器如何管理事务,
事务的属性控制了事务的使用范围,因为事务之间的关系非常的复杂,这个属性主要是用来处理事务与事务之间怎样来处理的的问题。
具体作用见附件图片。

如果产生一个系统异常,容器将自动回滚该事务。 EJBException是RuntimeException的子类,即也是一个系统运行时异常。
如果Bean抛出一个普通的非继承自RutimeException应用异常,事务将不会自动回滚,但可以
通过调用EJBContext的SetRollbackOnly回滚。
当然EJB上下文还有一个getRollbackOnly方法,通过返回一个boolean值来确定CMT事务是否已被标记为回滚。如果开始非常耗资源的操作前判断此bean的事务已被标记为回滚,则可以节约很多系统资源。
对于上面的异常回滚操作,还有一更加优雅的方式:


Java代码 收藏代码

1. public void placeSnagItOrder(Item item, Customer customer)
2. throws CreditValidationException, CreditProcessingException,
3. DatabaseException {
4. if (!bidsExisting(item)){
5. validateCredit(customer);
6. chargeCustomer(customer, item);
7. removeItemFromBidding(item);
8. }
9. }
10. ...
11. @ApplicationException(rollback=true)
12. public class CreditValidationException extends Exception {
13. ...
14. @ApplicationException(rollback=true)
15. public class CreditProcessingException extends Exception {
16. ...
17. //系统异常
18. @ApplicationException(rollback=false)
19. public class DatabaseException extends RuntimeException {

public void placeSnagItOrder(Item item, Customer customer)
throws CreditValidationException, CreditProcessingException,
DatabaseException {
if (!bidsExisting(item)){
validateCredit(customer);
chargeCustomer(customer, item);
removeItemFromBidding(item);
}
}
...
@ApplicationException(rollback=true)
public class CreditValidationException extends Exception {
...
@ApplicationException(rollback=true)
public class CreditProcessingException extends Exception {
...
//系统异常
@ApplicationException(rollback=false)
public class DatabaseException extends RuntimeException {


@ApplicationException把JAVA核对与不核对的异常标识为应用程序异常。其rollback默认为false,表示程序不会导致CMT自动回滚。
但应用程序异常应当慎用,见下文:

If the container detects a system exception, such as an ArrayIndexOutOfBounds or
NullPointerException that you did not guard for, it will still roll back the CMT transaction.
However, in such cases the container will also assume that the Bean is in inconsistent state and will
destroy the instance. Because unnecessarily destroying Bean instances is costly, you should never
delibertely use system exceptions.

2.bean管理事务

由于CMT依靠容器开始、提交和回滚事务,所以会限制事务的边界位置。而BMT则允许通过编程的方式来指定事务的开始、提交和回滚的位置。主要使用的是javax.transaction.UserTransaction接口。

如下面代码:

Java代码 收藏代码

1. @Stateless)
2. @TransactionManagement(TransactionManagementType.BEAN)
3. public class OrderManagerBean {
4. @Resource
5. private UserTransaction userTransaction;
6.
7. public void placeSnagItOrder(Item item, Customer customer){
8. try {
9. userTransaction.begin();
10. if (!bidsExisting(item)){
11. validateCredit(customer);
12. chargeCustomer(customer, item);
13. removeItemFromBidding(item);
14. }
15. userTransaction.commit();
16. } catch (CreditValidationException cve) {
17. userTransaction.rollback();
18. } catch (CreditProcessingException cpe){
19. userTransaction.rollback();
20. } catch (DatabaseException de) {
21. userTransaction.rollback();
22. } catch (Exception e) {
23. e.printStackTrace();
24. }
25. }
26. }

@Stateless)
@TransactionManagement(TransactionManagementType.BEAN)
public class OrderManagerBean {
@Resource
private UserTransaction userTransaction;

public void placeSnagItOrder(Item item, Customer customer){
try {
userTransaction.begin();
if (!bidsExisting(item)){
validateCredit(customer);
chargeCustomer(customer, item);
removeItemFromBidding(item);
}
userTransaction.commit();
} catch (CreditValidationException cve) {
userTransaction.rollback();
} catch (CreditProcessingException cpe){
userTransaction.rollback();
} catch (DatabaseException de) {
userTransaction.rollback();
} catch (Exception e) {
e.printStackTrace();
}
}
}


@TransactionManagement(TransactionManagementType.BEAN) 指定了事务的类型为BMT,上面没有用到@TransactionAttribute,因为它只适用于CMT,在上面代码中,可以显示的指定事务的开始与提交,因此更加的灵活。上面最关键的地方是注入了UserTransaction资源。其中获得UserTransaction资源的方式有三种,除了用 EJB资源的方式注入以外,还有以下两种方式:
(1) JNDI查找
Java代码 收藏代码

1. Context context = new InitialContext();
2. UserTransaction userTransaction =
3. (UserTransaction) context.lookup(“java:comp/UserTransaction”);
4. userTransaction.begin();
5. // Perform transacted tasks.
6. userTransaction.commit();

Context context = new InitialContext();
UserTransaction userTransaction =
(UserTransaction) context.lookup(“java:comp/UserTransaction”);
userTransaction.begin();
// Perform transacted tasks.
userTransaction.commit();


如果在EJB之外,则可使用此方法,如在不支持依赖注入的JBoss4.2.2的Web工程或helper class即帮助器类中都可以。

(2)EJBContext

Java代码 收藏代码

1. @Resource
2. private SessionContext context;
3. ...
4. UserTransaction userTransaction = context.getUserTransaction(); userTransaction.begin();
5. // Perform transacted tasks.
6. userTransaction.commit();

@Resource
private SessionContext context;
...
UserTransaction userTransaction = context.getUserTransaction(); userTransaction.begin();
// Perform transacted tasks.
userTransaction.commit();


注意:getUserTransaction方法只能在BMT中使用。如果在CMT中使用,则会抛出IllegalStateException 异常。且在BMT中不能使用EJBContext的getRollbackOnly和setRollbackOnly方法,如果这样使用,也会抛出 IllegalStateException异常。


如果使用有状态的Session Bean且需要跨越方法调用维护事务,那么BMT是你唯一的选择,当然BMT这种技术复杂,容易出错,且不能连接已有的事务,当调用BMT方法时,总会暂停已有事务,极大的限制了组件的重用。故优先考虑CMT事务管理。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值