1

我使用 JPA 并遇到以下问题,我的表在调用 setter 后会自动更新。这是一个例子:

持久对象:

@Entity
public class Product implements Serializable {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private String id;

    String name;

    /*+ getters & setters */

}

门面访问我的对象:

@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>

谢谢您的帮助。

4

2 回答 2

4

该规范要求管理从 EntityManager 返回的实体,以便它们可以跟踪更改并为您维护对象身份。

最简单的规范解决方案是不在事务中包装 EntityManager(如果使用 JTA,请确保非 JTA 数据源可用于读取),并关闭或清除 EntityManager。仅当 EntityManager 与事务关联时,更改才会同步到数据库,因此您的更改不能不持久化到数据库。只要您使用相同的 EntityManager,通过它的查询将返回相同的实体实例 - 因此您的更改将被拾取,直到您分离或清除 EntityManager,无论是否处于事务中。这就是您可能希望在逻辑点清除或关闭并重新获取 EntityManager 的原因。

在 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)
于 2013-09-06T18:15:16.633 回答
1

你必须分离你的实体。entityManager.detach()

于 2013-09-06T10:23:54.777 回答