2011-11-12 90 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>