1

我有三个表 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)超出了我们的事务范围。我想知道,如何在将持久性管理器注入事务范围时使事务完全发生?

4

1 回答 1

0

有关详细信息,请参阅下面的链接。

https://developer.jboss.org/message/525521

于 2015-10-30T11:44:21.243 回答