【问题标题】:JPA disable auto saveJPA 禁用自动保存
【发布时间】:2014-11-17 01:35:48
【问题描述】:

我使用 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();
    }

}

道播放器

@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 中分离实体?

【问题讨论】:

    标签: java jpa netbeans ejb


    【解决方案1】:

    理解这一点:“附加实体中的每次更新都会反映在数据库中”。

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

    请看下面的示例:

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

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

    如果您使用 EJB,您必须知道默认事务是必需的。 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 选项。 [=

    【讨论】:

    • 没有通用的方法来禁用自动保存的这种行为吗?我不想错过事务上下文,只想显式保存更改
    • 好吧,你应该使用 Bean Manager Transaction。您将手动控制交易。
    【解决方案2】:

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

    Player findById(Object id){}
    

    如果您在 Java EE 环境中使用容器管理事务,您只需将该事务标记为 DAO 中不支持。

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

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

    【讨论】:

    • 我应该在哪里放置注释先生?在实现类的接口中还是在 GenericDao 中?我已经在界面中测试了不起作用,但我会再次检查
    • 在实现类中。
    【解决方案3】:

    2021 年解决方案,

    把它放在班级的首位

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

    或者你可以使用,只在你的方法上,

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

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2011-04-25
      • 1970-01-01
      • 2015-05-10
      • 1970-01-01
      • 1970-01-01
      • 2021-11-20
      • 2012-10-03
      • 1970-01-01
      相关资源
      最近更新 更多