JPA禁用自动保存



i使用JPA eclipse Toplink&EJB

每次我从数据库接收实体,然后更改其中的一些数据,

数据库(MYSQL)也会自动更改,而无需我提交或做一些

示例:

Player p = em.findById(1);
p.setName(newName);
// i do nothing , but database automaticly change

我已经阅读了这里的相关问题,但我仍然无法弄清楚

我想我必须分离实体,但我不知道如何做到这一点,因为

我有从netbeans 生成的类GenericDAO

GenericDAO(从netbeans生成)

public abstract class GenericDAO<T> {
    private Class<T> entityClass;
    public GenericDAO(Class<T> entityClass) {
        this.entityClass = entityClass;
    }
    protected abstract EntityManager getEntityManager();
    public void insert(T entity) {
        getEntityManager().persist(entity);
    }
    public void edit(T entity) {
        getEntityManager().merge(entity);
    }
    public void delete(T entity) {
        getEntityManager().remove(getEntityManager().merge(entity));
    }
    public T findById(Object id) {
        return getEntityManager().find(entityClass, id);
    }
    public List<T> findAll() {
        javax.persistence.criteria.CriteriaQuery cq = getEntityManager().getCriteriaBuilder().createQuery();
        cq.select(cq.from(entityClass));
        return getEntityManager().createQuery(cq).getResultList();
    }
    public List<T> findRange(int[] range) {
        javax.persistence.criteria.CriteriaQuery cq = getEntityManager().getCriteriaBuilder().createQuery();
        cq.select(cq.from(entityClass));
        javax.persistence.Query q = getEntityManager().createQuery(cq);
        q.setMaxResults(range[1] - range[0] + 1);
        q.setFirstResult(range[0]);
        return q.getResultList();
    }
    public int count() {
        javax.persistence.criteria.CriteriaQuery cq = getEntityManager().getCriteriaBuilder().createQuery();
        javax.persistence.criteria.Root<T> rt = cq.from(entityClass);
        cq.select(getEntityManager().getCriteriaBuilder().count(rt));
        javax.persistence.Query q = getEntityManager().createQuery(cq);
        return ((Long) q.getSingleResult()).intValue();
    }
}

DaoPlayer

@Local
public interface DaoPlayer{
    void insert(Player player);
    void edit(Player player);
    void delete(Player player);
    Player findById(Object id);
    List<Player> findAll();
    List<Player> findRange(int[] range);
    int count();

ImplPlayer

@Stateless
public class ImplPlayer extends GenericDAO<Player> implements DaoPlayer{
    @PersistenceContext(unitName = "MonsterPuzzle")
    private EntityManager em;
    @Override
    protected EntityManager getEntityManager() {
        return em;
    }
    public ImplPlayer() {
        super(Player.class);
    }
}

我的问题是

  • 如何禁用JPA自动保存到数据库
  • (因为我认为我需要分离)如何将实体从GenericDAO分离

理解这一点:"附加实体中的每个更新都将反映在数据库中"。

何时附着实体?简单地说,每个通过EntityManager的实体都有以下方法:持久化、合并、刷新、删除(也许我在这里忘记了其他方法)或事务中的查询(JPQL或find)方法。

看看下面的例子:

entityManager.getTransaction().begin();
Dog dog = entityManager.find(Dog.class, 1);
dog.setGoodBoyLevel(99);
entityManager.getTransaction().commit();

由于dog在事务中通过了find方法,因此可以认为它是托管的。

如果您正在使用EJB,您必须知道默认事务是REQUIRED。REQUIRED的工作原理是:如果有一个交易打开了,让我们使用它。如果没有交易打开,让我们打开它。(这是一个好人)

由于您的方法没有任何挂起事务的指示符,因此实体中的每个更新都将反映在数据库中。你能做什么?使用NOT_SUPPORTED交易方式:

@TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED)
public void yourMethod(){
    // to awesome things
}

由于您使用NOT_SUPPORTED事务选项,您的实体将被分离分离意味着什么?JPA不会分析实体中的更新。如果你只想阅读一些信息,那没关系。但下面的代码会引发异常:

entityManager.getTransaction().begin();
Dog dog = new Dog();
Owner owner = getDetachedOwner(ownerId);
dog.setOwner(owner);
entityManager.persist(dog);
entityManager.getTransaction().commit();

这里的问题是所有者是分离的,所以JPA不知道这一点,并且会引发一个异常。

注意NOT_SUPPORTED选项。[=

您需要了解如何在应用程序中管理事务。考虑一下你的一种方法:

Player findById(Object id){}

如果您在Java EE环境中使用Container Managed Transactions,则只需要将该事务标记为DAO中不支持该事务。

@TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED)
Player findById(Object id) {}

这将确保当Dao将实体传递回另一层时,事务不会打开。对实体所做的任何更改都不会与数据库同步。

2021解决方案,

把它放在你班的首位

@Transactional(propagation = Propagation.NOT_SUPPORTED)
 public class YourClass{
   ...
} 

或者你只能在你的方法上使用

@Transactional(propagation = Propagation.NOT_SUPPORTED)
 public void yourMethod(){
   ...
} 

相关内容

  • 没有找到相关文章

最新更新