2014-10-28 98 views
5

我有一個相當簡單的問題。在一個單一的交易,我的代碼看起來是這樣的:Spring數據JPA JpaRepository.save(實體)不返回數據庫默認值

MyClass c = new MyClass(); 
c.setPropA("A"); 

c = myClassRepository.save(c); 

c.setPropC("C"); 

在哪裏我的實體是這樣的:

@Entity 
@Table(name = "MY_CLASS") 
public class MyClass { 
    private String propA; 
    private String propB; 
    private String propC; 

    @Id 
    @Column(name = "PROP_A", unique = true, nullable = false, updatable = false) 
    public String getPropA() { return propA; } 
    public void setPropA(String propA) { this.propA = propA; } 

    @Column(name = "PROP_B", insertable = false) 
    public String getPropB() { return propB; } 
    public void setPropB(String propB) { this.propB = propB; } 

    @Column(name = "PROP_C") 
    public String getPropC() { return propC; } 
    public void setPropC(String propC) { this.propC = propC; } 
} 

和數據庫建立這樣的(方言=甲骨文11G)

CREATE TABLE MY_CLASS { 
    PROP_A VARCHAR2 NOT NULL PRIMARY KEY, 
    PROP_B VARCHAR2 DEFAULT 'B' NOT NULL, 
    PROP_C VARCHAR2 
} 

因此,基於堅持實體的代碼,我想我會得到這樣的查詢:

MyClass c = new MyClass(); 
c.setPropA("A"); 

c = myClassRepository.saveAndFlush(c); 
// INSERT INTO MY_CLASS (PROP_A, PROP_C) VALUES ("A", NULL) 
// SELECT PROP_A, PROP_B, PROP_C FROM MY_CLASS -- to refresh `c` 

c.setPropC("C"); 

// After transaction ends, flush change to `c.propC` 
// UPDATE MY_CLASS SET PROP_B = "B", PROP_C = "C" WHERE PROP_A = "A" 

...但SELECT從未發生。爲什麼實體不刷新?

由於UPDATE(立即)跟在INSERTc.propB仍然是null,因此中斷。我得到一個ORA-01407: cannot update ("MY_SCHEMA"."MY_CLASS"."PROP_B") to NULL堆棧跟蹤。

我知道有些方法可以使用@PrePersist或在Java中設置默認值columnDefinition,但我不應該重複默認值。

注意:在我使用org.hibernate.Session.saveOrUpdate()代替Spring Data JPA進行此操作之前,此工作方式有效。

在此先感謝您的幫助!

回答

4

Hibernate不知道B是由數據庫生成的,並且在插入後必須讀取它,除非您使用@Generated annotation告訴它這樣做。

+0

好抓! '@ GeneratedValue'是JPA的等價物嗎? – Andy 2014-10-30 13:46:36

+0

在我的情況下,我有一列'modified_at',默認值是當前日期時間。唯一對我有用的是[@UpdateTimestamp](https://docs.jboss.org/hibernate/orm/4.3/javadocs/org/hibernate/annotations/UpdateTimestamp.html)。我不需要調用'.flush()'。 既不[@Generated](http://docs.jboss.org/hibernate/orm/4.3/javadocs/org/hibernate/annotations/Generated.html)或[@GeneratedValue](http://docs.oracle.com)。 com/javaee/5/api/javax/persistence/GeneratedValue.html),即使在使用'.flush()'後,該字段仍會在'.save()'後返回爲空。 – Doug 2017-10-04 13:14:00

相關問題