我使用JPA并遇到以下问题,我的表在调用setter后会自动更新。 这是一个例子:
持久对象:
@Entity
public class Product implements Serializable {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private String id;
String name;
/*+ getters & setters */
}
Facade访问我的对象:
@Stateless
public class ProductFacade {
@PersistenceContext(unitName = "PU")
private EntityManager em;
public Product find(Object id) {
Product p = em.find(Product.class, id);
return p;
}
public void create(Product p) {
em.persist(p);
}
}
但是当我在我的webapp中执行此代码时:
@Stateless
public class ProductService {
@EJB
ProductFacade manager;
public void testMethod() {
id = new Long(1);
Product product = manager.find(id);
product.setName("newName");
}
}
产品的名称值在设置后在数据库中更改, 即使我没有明确要求保持或合并。
如何避免这种自动数据库更改?
这也是我的persistence.xml:
<persistence .... >
<persistence-unit name="PU" transaction-type="JTA">
<provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
<jta-data-source>jdbc/sample</jta-data-source>
<exclude-unlisted-classes>false</exclude-unlisted-classes>
<properties>
<property name="eclipselink.ddl-generation" value="create-tables"/>
<property name="connection.autocommit" value="false"/>
<property name="eclipselink.persistence-context.flush-mode" value="COMMIT"/>
</properties>
</persistence-unit>
</persistence>
感谢您的帮助。
答案 0 :(得分:4)
规范要求管理从EntityManager返回的实体,以便他们可以跟踪更改并维护对象标识。
最简单的规范解决方案是不将EntityManager包装在事务中(如果使用JTA,请确保非JTA数据源可用于读取),并关闭或清除EntityManager。如果EntityManager与事务关联,则更改将仅同步到数据库,因此您的更改不能持久保存到数据库。只要您使用相同的EntityManager,通过它进行的查询将返回相同的实体实例 - 因此,在您分离或清除EntityManager之前,无论是否处于事务中,您的更改都将被选中。这就是您可能希望在逻辑点清除或关闭并重新获取EntityManagers的原因。
在EclipseLink中,您可以使用提示标记查询,以告知EclipseLink返回分离的对象。有关信息,请参阅http://wiki.eclipse.org/EclipseLink/UserGuide/JPA/Basic_JPA_Development/Caching/Query_Options#EclipseLink_Cache_Query_Hints中描述的eclipselink.maintain-cache提示。它也可以在http://wiki.eclipse.org/EclipseLink/UserGuide/JPA/Basic_JPA_Development/Query_Hints#Maintain_Cache找到,但指定的默认值不正确 - 默认值为true。
query.setHint(QueryHints.MAINTAIN_CACHE, false);
或
Map properties = new HashMap();
properties.put(QueryHints.MAINTAIN_CACHE, false);
em.find(Product.class, id, properties)
答案 1 :(得分:1)
您必须分离您的实体。 entityManager.detach()