2016-11-10 46 views
1

我正在Hibernate 4.3.8上使用Web应用程序,它看起来像persist()方法不会更新PersistentContext(高速缓存级别1)。下面是我的配置和单管理持久性的操作:Hibernate persist()不更新PersistentContext中的模型

Hibernate配置

<persistence-unit name="PersistenceUnit" transaction-type="RESOURCE_LOCAL"> 
    <provider>org.hibernate.jpa.HibernatePersistenceProvider</provider> 
    <properties> 
     <property name="hibernate.dialect" value="org.hibernate.dialect.PostgreSQLDialect"/> 
     <property name="hibernate.connection.driver_class" value="org.postgresql.Driver"/> 
     <property name="hibernate.connection.url" value="jdbc:postgresql://localhost:5432/irm"/> 
     <property name="hibernate.connection.username" value="nrossi"/> 
     <property name="hibernate.connection.password" value="nicolas"/> 
     <property name="hibernate.connection.pool_size" value="5"/> 
     <property name="hibernate.show_sql" value="true"/> 
     <property name="hibernate.format_sql" value="true"/> 
     <property name="hibernate.max_fetch_depth" value="5"/> 
     <property name="hibernate.hbm2ddl.auto" value="update"/> 
    </properties> 
</persistence-unit> 

持久性管理

public class PersistenceManager 
{ 
    private static final Logger log = LogManager.getLogger(); 
    private static final EntityManagerFactory emf; 
    private static final ThreadLocal<EntityManager> threadLocal; 

    static 
    { 
     emf = Persistence.createEntityManagerFactory("PersistenceUnit"); 
     threadLocal = new ThreadLocal<EntityManager>(); 
    } 

    public static EntityManager getEntityManager() 
    { 
     EntityManager em = threadLocal.get(); 

     if (em == null) 
     { 
      em = emf.createEntityManager(); 
      threadLocal.set(em); 
     } 
     return em; 
    } 

    public static <T>T get(Class<T> clazz, Object id) 
    { 
     return getEntityManager().find(clazz, id); 
    } 

    public static void save(Object object) 
    { 
     EntityManager em = getEntityManager(); 
     EntityTransaction et = em.getTransaction(); 
     et.begin(); 
     try 
     { 
      em.persist(object); 
      et.commit(); 
     } 
     catch (Exception e) 
     { 
      et.rollback(); 
      throw new RuntimeException("Error saving object", e); 
     } 
    } 
} 

我更新模型调用PersistenceManager.save(model)并更新记录在数据库上,但在此之后,当我调用PersistenceManager.get(model.id)时,它将使用旧值从内存返回模型。它看起来像persist方法没有更新PersistenceCache。

顺便说一下,如果我在新线程(即隐身窗口)上调用PersistenceManager.get(model.id),它将返回更新的模型。

我试着在提交之后添加一个刷新调用em.refresh(model)并且它正在工作,但我不确定这是否是更新上下文的正确方法。

更新信息 我编写了一个简单的JSP页面来重现行为。如果我更新的实体描述,并等待5' 和刷新页面,则返回旧值:

<%@page import="com.identicum.framework.persistence.PersistenceManager"%> 
<%@page import="com.identicum.irm.core.model.Entity"%> 
<% 
    Entity entity = PersistenceManager.get(Entity.class, 1L); 
    String value = request.getParameter("entityName"); 
    if(value != null) 
    { 
     entity.setDescription(value); 
     PersistenceManager.save(entity); 
    } 
%> 
<html> 
<body> 

Entity description: <b><%= entity.getDescription() %></b> 

<br> 
<br> 

<form method="post"> 

    Enter new entity description <br> 
    <input type="text" name="entityName"/> 

    <input type="submit" /> 

</form> 


</body> 
</html> 

**新信息**

我的PersistenceManager这是suggestion的副本。我的应用程序是一个应用程序管理的EntityManager。我必须在每个请求的生命周期中访问同一个EntityManager。这就是这种方法的原因。有没有其他的方式来实现这一点?

+0

后的所有相关代码,包括你在哪里加载和更新这些实体。 –

+0

有其处理每个请求并运行下面的命令一个servlet: 1)为了获得模型:'PersistenceManager.get(Entity.class,ID);'和 2)更新模型: 实体实体= PersistenceManager.get(Entity.class,id); entity.setName(“B”); PersistenceManager.save(entity); PersistenceManager代码是第一个问题。 – chupetoide

+0

嗨艾伦,我简化了代码,并写了一个简单的jsp来重现行为。你可以在这里找到它(https://dl.dropboxusercontent.com/u/9319179/test.jsp)。我发现问题在几分钟后开始。我运行jsp并向实体发布新描述。然后等待5分钟,我再次运行jsp(不带参数),然后再次看到第一个描述。 – chupetoide

回答

0

首先persist()用于新实体(数据库中没有记录)。 如果你想改变一个对象,你必须使用merge()。

但是,你的问题是,你可能是在一个多线程环境。 所以你不能使用ThreadLocal。每个线程都有他自己的PersistenceManager。

您使用的是什么应用程序服务器或Web容器?

+0

这是一个Tomcat应用服务器。但问题是它看起来像新线程从PersistentContext获取旧模型。我不确定问题是否在每个线程上都有自己的PersistenceManager。在这种情况下,我可能会遇到“性能问题”,但不是“缓存问题”。你同意吗? – chupetoide

+0

我不认为你会遇到性能问题。在Java EE服务器中,PersistenceContext绑定到事务。使用merge(),在更新之前不必读取实体。 –

+0

我知道,但即使当我使用合并而不是持续时返回对象时,问题仍然是缓存 – chupetoide

0

我终于修好了,改变了我得到的方式并关闭了EntityManager。现在我从每个请求开始时(在我的Rest Dispatcher上)从EntityManagerFactory获取EntityManager,并在最后关闭它。这样每个请求都有自己的PersistentContext。

PersistenceMangaer.java几乎相同。主要区别在于我删除了EntityManagerFactory对ApplicationServiceContext的引用,我在课程外部调用setEntityManager()。这里有一个例子:

PersistenceManager.setEntityManager(PersistenceFactory.createEntityManager); 
FooService.createObject1(); 
FooService.createObject2(); 
PersistenceManager.closeEntityManager(); 

这些链接,其中有利于发现问题:

Best practice to get EntityManagerFactory

Should JPA Entity Manager be closed?

http://www.objectdb.com/java/jpa/start/connection