我有三个表 server_detail、server_group、server_group_mapping 和实体类如下。(没有给出完整的代码细节)
@Entity
@Table(name = "server_detail")
public class ServerBean implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "server_id")
private Integer ServerId;
@Column(name = "name")
private String serverName;
....
}
@Entity
@Table(name = "server_group")
public class ServerGroupBean implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "group_id")
private Integer groupId;
@Column(name = "name")
private String groupName;
....
}
@Entity
@IdClass(GroupMapPK.class)
@Table(name = "server_group_mapping")
public class ServerGroupMapBean implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@Column(name = "group_id")
private Integer groupId;
@Id
@Column(name = "server_id")
private Integer serverId;
....
}
每个实体 Bean 类都有一个包装类,用于管理实体 bean 上的操作,如下所示
@Stateless
@Local
public class ServerClient implements ServerLocal {
@PersistenceContext
private EntityManager em;
public ServerClient() {
}
public ServerBean create(java.lang.String name) {
ServerBean bean = new ServerBean(name);
em.persist(bean);
return bean;
}
public ServerBean update(ServerBean bean){
return (em.contains(bean) ? bean : em.merge(bean));
}
public void remove(ServerBean bean) {
em.remove(em.contains(bean) ? bean : em.merge(bean));
}
@Stateless
@Local
public class ServerGroupClient implements ServerGroupLocal {
@PersistenceContext
private EntityManager em;
public ServerGroupClient() {
}
public ServerGroupBean create(java.lang.Integer name) {
ServerGroupBean bean = new ServerGroupBean(name);
em.persist(bean);
return bean;
}
public ServerGroupBean update(ServerGroupBean bean){
return (em.contains(bean) ? bean : em.merge(bean));
}
public void remove(ServerGroupBean bean) {
em.remove(em.contains(bean) ? bean : em.merge(bean));
}
@Stateless
@Local
public class ServerGroupMapClient implements ServerGroupMapLocal {
@PersistenceContext
private EntityManager em;
public ServerGroupMapClient() {
}
public ServerGroupMapBean create(java.lang.Integer serverId,java.lang.Integer groupId ) {
ServerGroupMapBean bean = new ServerBean(serverId, groupId);
em.persist(bean);
return bean;
}
public ServerBean update(ServerBean bean){
return (em.contains(bean) ? bean : em.merge(bean));
}
public void remove(ServerBean bean) {
em.remove(em.contains(bean) ? bean : em.merge(bean));
}
我对表使用 MYSQL(innoDB 引擎)以及它们在表中的关系映射。
现在,我有 GroupManager 会话 Bean 类,它维护 server_group 和 server_group_mapping 表事务。每当我创建服务器组和成员时,我都必须进行以下交易。
1. First, add group id and group name to server_group table
2. Second, map groupid with server id in server_group_mapping table
以下是代码。
@Stateless
@Local
public class GroupManagerBean implements GroupManagerLocal {
@Resource
private SessionContext context;
private static GroupLocal GroupLocal;
private static GroupMapLocal GroupMapLocal;
public GroupManagerBean() {
GroupLocal = ServiceLocator.getGroupLocal();
smscMapLocal = ServiceLocator.getGroupMapLocal();
}
public void addGroup(GroupBean bean, Integer serverId){
group = GroupLocal.create(bean.getGroupName()); ---> 1
...
GroupMapLocal.create(group.getGroupId(), serverId); ----> 2
}
ServiceLocator 类是我可以查找所有 bean 的位置。默认情况下,在 ejb3 中,事务属性是必需的。如果我执行 addGroup() 方法。我得到以下异常。
javax.ejb.EJBTransactionRolledbackException: EntityManager must be access within a transaction
at org.jboss.ejb3.tx.Ejb3TxPolicy.handleInCallerTx(Ejb3TxPolicy.java:115)
at org.jboss.aspects.tx.TxPolicy.invokeInCallerTx(TxPolicy.java:130)
at org.jboss.aspects.tx.TxInterceptor$Required.invoke(TxInterceptor.java:194)
at org.jboss.aop.joinpoint.MethodInvocation.invokeNext(MethodInvocation.java:102)
at org.jboss.aspects.tx.TxPropagationInterceptor.invoke(TxPropagationInterceptor.java:76)
at org.jboss.aop.joinpoint.MethodInvocation.invokeNext(MethodInvocation.java:102)
at org.jboss.ejb3.tx.NullInterceptor.invoke(NullInterceptor.java:42)
at org.jboss.aop.joinpoint.MethodInvocation.invokeNext(MethodInvocation.java:102)
at org.jboss.ejb3.security.Ejb3AuthenticationInterceptorv2.invoke(Ejb3AuthenticationInterceptorv2.java:186)
at org.jboss.aop.joinpoint.MethodInvocation.invokeNext(MethodInvocation.java:102)
at org.jboss.ejb3.ENCPropagationInterceptor.invoke(ENCPropagationInterceptor.java:41)
at org.jboss.aop.joinpoint.MethodInvocation.invokeNext(MethodInvocation.java:102)
at org.jboss.ejb3.asynchronous.AsynchronousInterceptor.invoke(AsynchronousInterceptor.java:106)
at org.jboss.aop.joinpoint.MethodInvocation.invokeNext(MethodInvocation.java:102)
at org.jboss.ejb3.BlockContainerShutdownInterceptor.invoke(BlockContainerShutdownInterceptor.java:67)
at org.jboss.aop.joinpoint.MethodInvocation.invokeNext(MethodInvocation.java:102)
at org.jboss.aspects.currentinvocation.CurrentInvocationInterceptor.invoke(CurrentInvocationInterceptor.java:67)
at org.jboss.aop.joinpoint.MethodInvocation.invokeNext(MethodInvocation.java:102)
at org.jboss.ejb3.session.SessionSpecContainer.invoke(SessionSpecContainer.java:219)
at org.jboss.ejb3.proxy.handler.ProxyInvocationHandlerBase.invoke(ProxyInvocationHandlerBase.java:261)
....
Caused by: javax.persistence.TransactionRequiredException: EntityManager must be access within a transaction
at org.jboss.jpa.deployment.ManagedEntityManagerFactory.verifyInTx(ManagedEntityManagerFactory.java:155)
at org.jboss.jpa.tx.TransactionScopedEntityManager.persist(TransactionScopedEntityManager.java:186)
at com.example.GroupClient.create(GroupClient.java:37)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
根据错误持久性管理器(GroupClient 和 GroupMapClient)超出了我们的事务范围。我想知道,如何在将持久性管理器注入事务范围时使事务完全发生?