当前位置: 首页 > 面试题库 >

Spring Data JPA JpaRepository.save(entity)不返回数据库默认值

陶高峻
2023-03-14
问题内容

我有一个很简单的问题。在单个事务中,我的代码如下所示:

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; }
}

像这样建立数据库(Dialect = Oracle 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堆栈跟踪。

我意识到有很多方法可以在Java中使用@PrePersist或设置默认值columnDefinition,但是我不必重复默认值。

注意:这在我org.hibernate.Session.saveOrUpdate()代替Spring Data JPA进行此操作之前有效。

在此先感谢您的帮助!


问题答案:

Hibernate不知道B是由数据库生成的,并且它必须在插入后读取它,除非您使用@Generated批注指示它这样做。



 类似资料:
  • 问题内容: 我知道一个问题已经得到广泛讨论,但是我认为仍有一个方面需要澄清。 我正在创建一个具有多语言数据库的Web应用程序,我已经找到了一些很好的实践文章(例如 this), 因此,我决定使用一个包含项目ID的主表,以及一个包含每个项目的翻译的表,例如 要么 等等。 现在我在做什么?我只是从数据库中获取所有翻译的项目,然后遍历每个翻译,以根据当前用户的本地来查找正确的翻译,如果找到正确的本地,则

  • 问题内容: 我希望模仿允许用户指定“默认”返回值的内置函数(如)的行为。我最初的尝试是这样做 问题是,如果用户希望成为其返回值,则此函数将引发异常。第二次尝试: 这解决了上述问题,并允许用户指定任何任意值,但是带来了麻烦,因为用户必须始终在其函数调用中进行指定;他们不能只提供最后。同样,可以使用,但如果用户偏爱该语法,则会阻止用户使用。 结合并提供了可行的解决方案,但感觉这需要付出很多努力。它还可

  • 我有一个相当简单的问题。在单个事务中,我的代码如下所示: 我的实体看起来像这样: 数据库是这样设置的(方言=Oracle 11G) 因此,基于持久化实体的代码,我想我会得到如下查询: ... 但是永远不会发生。为什么实体没有刷新? 这是因为

  • 在我的quarkus应用程序中有两个命名的数据源,我们称之为A和B: Quarkus.datasource.a.db-kind=Oracle Quarkus.datasource.b.db-kind=Oracle 因此,每次应用程序启动时,我都会收到这样的通知: 警告:无法确定默认数据源的数据库类型

  • 问题内容: 我有以下选择语句,以获取流的下一个预定项目。如果没有匹配的行,我希望它返回默认值。这是我正在使用的行: 那应该可以获取最近计划的项目,但如果它早于查询之前30分钟,则不会。但是,如果用户未安排任何时间,则我需要一个默认值,以便流中实际播放某些内容。我尝试了以下方法: 和: 但是,如果未找到任何行,它将始终返回空结果。我该如何返回默认值? 问题答案: 一种方法 由于只返回一行,因此可以使

  • 安装成功后系统会提示面板访问地址与AMH、MySQL默认账号密码。 这是官方进行极速安装的时候显示的一段话。 为什么软件的默认数据库选择MYSQL。其实可以和宝塔那样。软件本身的数据库是独立的。 这样更有利于系统的安装。MYSQL数据库后期可以根据自己的需求进行安装。