我使用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);
}
}
我的问题是
答案 0 :(得分:3)
理解这一点:&#34; 附加实体中的每个更新都将反映在数据库中&#34;。
何时附加实体?换句话说,每个通过EntityManager传递方法的实体都有:持久化,合并,刷新,删除(也许我忘了其他任何一个)或者在事务中的查询(JPQL或find)方法。
请看下面的示例:
entityManager.getTransaction().begin();
Dog dog = entityManager.find(Dog.class, 1);
dog.setGoodBoyLevel(99);
entityManager.getTransaction().commit();
由于狗在事务中通过find方法传递,因此可以将其视为托管。
如果您使用的是EJB,则必须知道默认事务是必需的。必需的工作如下:如果有一个交易打开,让我们使用它。如果没有打开任何交易,请让我们打开它。 (这是一个好人)
由于您的方法没有任何指示符来暂停事务,因此实体中的每个更新都将反映在数据库中。你能做什么?使用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选项。 [=
答案 1 :(得分:2)
您需要了解如何管理应用程序中的事务。考虑一下你的方法:
Player findById(Object id){}
如果您在Java EE环境中使用容器管理事务,则只需将事务标记为DAO不支持。
@TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED)
Player findById(Object id) {}
当Dao将实体传回另一层时,这将确保事务未打开。您对实体所做的任何更改都不会与数据库同步。
答案 2 :(得分:-1)
2021 年解决方案,
<块引用>把它放在你的班级中
@Transactional(propagation = Propagation.NOT_SUPPORTED)
public class YourClass{
...
}
<块引用>
或者你可以只使用你的方法,
@Transactional(propagation = Propagation.NOT_SUPPORTED)
public void yourMethod(){
...
}