首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >禁用自动保存

禁用自动保存
EN

Stack Overflow用户
提问于 2014-09-23 00:23:51
回答 3查看 7.7K关注 0票数 7

我使用JPA、和EJB

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

数据库(MYSQL)也会自动更改,而不需要提交或做某事。

示例:

代码语言:javascript
复制
Player p = em.findById(1);
p.setName(newName);
// i do nothing , but database automaticly change

我已经在这里读过相关的问题了,但我还是搞不懂

我想我必须分离实体,但我不知道怎么做,因为

我有从netbeans生成的类GenericDAO

GenericDAO (由netbeans生成)

代码语言:javascript
复制
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

代码语言:javascript
复制
@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

代码语言:javascript
复制
@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中分离实体?
EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2014-09-23 00:36:38

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

代码语言:javascript
复制
Player findById(Object id){}

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

代码语言:javascript
复制
@TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED)
Player findById(Object id) {}

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

票数 1
EN

Stack Overflow用户

发布于 2014-09-23 02:03:52

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

当一个实体被附加的时候?简单地说,每一个通过EntityManager通过方法的实体:持久化、合并、刷新、删除(可能这里我忘记了任何其他方法),或者在事务中的查询(JPQL或find)方法中。

请看下面的示例:

代码语言:javascript
复制
entityManager.getTransaction().begin();
Dog dog = entityManager.find(Dog.class, 1);
dog.setGoodBoyLevel(99);
entityManager.getTransaction().commit();

因为狗在事务中通过find方法传递,所以它可以被认为是管理的。

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

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

代码语言:javascript
复制
@TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED)
public void yourMethod(){
    // to awesome things
}

由于您使用的是NOT_SUPPORTED事务选项,所以您的实体将被分离。分离意味着什么?联合行动纲领将不分析实体中的最新情况。如果你只想读一些信息,那就好了。但下面的代码将引发例外情况:

代码语言:javascript
复制
entityManager.getTransaction().begin();
Dog dog = new Dog();
Owner owner = getDetachedOwner(ownerId);
dog.setOwner(owner);
entityManager.persist(dog);
entityManager.getTransaction().commit();

这里的问题是所有者是独立的,所以JPA不知道它,因此会引发异常。

小心使用NOT_SUPPORTED选项。[=

票数 2
EN

Stack Overflow用户

发布于 2021-01-12 13:38:54

2021解决方案,

把它放在班上的第一位

代码语言:javascript
复制
@Transactional(propagation = Propagation.NOT_SUPPORTED)
 public class YourClass{
   ...
} 

或者你只能用你的方法,

代码语言:javascript
复制
@Transactional(propagation = Propagation.NOT_SUPPORTED)
 public void yourMethod(){
   ...
} 
票数 -1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/25985047

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档