EntityManager不会写入数据库

时间:2011-11-12 16:07:08

标签: hibernate jpa hsqldb entitymanager

我刚建立了一个迄今为止仍然非常小的项目maven / jpa / hibernate项目,我试图坚持一个对象。

我的课很简单:

@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表(由于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数据库文件,但不会创建表。

其次,我希望持久化对象存储在数据库中,但事实并非如此。此外,手动创建数据库模式并不能解决问题,因此不会导致问题。持久化代码在输出中没有任何异常或任何警告的情况下运行,看起来一切都很好。但是这个对象并没有到达数据库。使用“from Person”查询实体管理器时也找不到该对象。

然而,查询似乎是唯一有效的方法。我可以手动将数据插入数据库,“from Person”查询将成功检索它。

那么,我在这里做错了什么提示?

3 个答案:

答案 0 :(得分: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 ms,需要睡眠(1000)。给出的信息适用于HSQLDB 2.2.x版。

答案 1 :(得分:3)

您需要将shutdown=true添加到数据库URL(或发出explict SHUTDOWN命令)才能正确关闭进程内HSQLDB数据库:

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

另见:

答案 2 :(得分:0)

希望这有助于某人。将写入延迟设置为0 ms的建议有效,但除非您正在创建新数据库,否则它无法从URL中运行。要使其适用,请在使用下面的sql语句连接到数据库时在脚本中运行它:

set files write delay false

例如,您可以使用sqltool.jar运行它,如下所示:

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