2011-11-12 93 views
5

我只是建立了一个迄今为止仍然非常小的项目maven/jpa/hibernate项目,我试图持久化一个对象。EntityManager不写入数据库

我的类是一个很简单的一个:

@Entity 
public class Person { 
    @Id @GeneratedValue 
    private int id; 
    private String name; 
} 

我的persistence.xml是非常基本的,以及:

<persistence xmlns="http://java.sun.com/xml/ns/persistence" 
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
    xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd" 
    version="2.0"> 
    <persistence-unit name="Fahrplan_v2"> 
     <class>model.Person</class> 
     <properties> 
      <property name="hibernate.connection.driver_class" value="org.hsqldb.jdbcDriver" /> 
      <property name="hibernate.connection.url" value="jdbc:hsqldb:file:data/db/db" /> 
      <property name="hibernate.connection.username" value="sa" /> 
      <property name="hibernate.connection.password" value="" /> 
      <property name="hibernate.dialect" value="org.hibernate.dialect.HSQLDialect" /> 
      <property name="hibernate.hbm2ddl.auto" value="update" /> 
     </properties> 
    </persistence-unit> 
</persistence> 

而且最后这里是我用来持久化对象的代码:

EntityManager em = entityManagerFactory.createEntityManager(); 
em.getTransaction().begin(); 
em.persist(person); 
// em.flush(); <- does not effect outcome. 
em.getTransaction().commit(); 
em.close(); 

现在有两件事情我希望在这里发生:首先,我期望Person表是cr eated(由于hibernate.hbm2ddl.auto = update)。这发生了一次,它正确地写出

CREATE MEMORY TABLE PUBLIC.PERSON(
    ID INTEGER GENERATED BY DEFAULT AS IDENTITY(START WITH 1) NOT NULL PRIMARY KEY, 
    NAME VARCHAR(255) 
) 

但我不能重现那。每次我开始我的程序时,都会创建hsqldb数据库文件,但不会创建表。

其次,我期望持久对象存储在数据库中,但事实并非如此。另外,手动创建数据库模式并不能解决问题,所以这不是导致它的原因。持久化代码在输出中运行时没有任何异常或任何警告,它看起来非常好。但是这个对象并没有到达数据库。使用“来自Person”查询实体经理时也找不到该对象。

然而,查询似乎是唯一能够工作的东西。我可以手动将数据插入数据库,“从人”查询将成功检索它。

所以,我在这里做错了什么暗示?

+1

呜呜好了,所以我会后的评论,因为我不能回答我自己的问题(没有?): 我发现我的问题。 em.getTransaction()提交(); Thread.sleep(1000); em.close(); 做到了。 在我的测试用例之后,JVM快速关闭,hsqldb没有足够的时间将提交的数据保存到磁盘。不要问我为什么JVM不活跃,我不做任何System.exit()调用或任何东西,我的junit测试方法只是在那里结束。出于某种原因,我认为JVM并不能保持活力,这是另一个应该坚持我的对象的线程。 – MisterD

+0

你被允许回答你的问题,事实上鼓励这样做。阅读常见问题。你的问题很好,所以每个人都可以很好地总结出答案。 – Siddharth

回答

5

添加到axtavt的良好答案并澄清您的睡眠方式(1000)如何工作:对于您希望完全同步持久性的开发情况,请关闭默认的write_delay。

<property name="hibernate.connection.url" 
     value="jdbc:hsqldb:file:data/db/db;shutdown=true;hsqldb.write_delay_millis=0"/>    

这意味着每个语句都会在结果返回给调用者之前写入磁盘。自然地,在正常操作中,您可以增加此值。默认值为500毫秒,需要睡眠(1000)。给出的信息用于HSQLDB 2.2.x版。

3

您需要添加shutdown=true到数据库URL(或发出显式SHUTDOWN命令),以便正确地关闭一个进程HSQLDB数据库:

<property name="hibernate.connection.url" 
    value="jdbc:hsqldb:file:data/db/db;shutdown=true" />    

参见:

+0

啊,是的,忘了提及那个,我之前有过,并且删除了它,因为我认为它可能会导致这个问题。它无法使用或不使用它。 :( – MisterD

0

希望这可以帮助别人。建议将写延迟设置为0毫秒,但它不起作用,除非你创建一个新的数据库。把它应用,在脚本中运行它,而连接到数据库中使用下面的SQL语句:

set files write delay false 

因此,例如,可以使用虽然SqlTool运行它。罐子像这样:

java -jar /path/to/sqltool.jar -sql "set files write delay false;" --inlinerc=<rc spec here>