Hibernate 5 中 GenerationType.AUTO 问题

当你使用 GenerationType.AUTO 时,Hibernate 会根据 Hibernate 方言生成策略。如果你需要支持多个数据库,以下一种常用的方法。

@Entity
public class Author {
 
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Column(name = "id", updatable = false, nullable = false)
    private Long id;
 
    ...
}

在旧版本中,Hibernate 为 MySQL 数据库选择 GenerationType.IDENTITY。这是一个不错的选择。如之前所述,这是最有效的方法。

不幸的是此行为在 Hibernate 5 中发生了改变,现在使用的是 GenerationType.TABLE,它使用数据库表来生成主键。这种方法需要大量数据库查询和悲观锁来生成唯一值,并且多个表将可能共享自增id导致混乱。

14:35:50,959 DEBUG [org.hibernate.SQL] - select next_val as id_val from hibernate_sequence for update
14:35:50,976 DEBUG [org.hibernate.SQL] - update hibernate_sequence set next_val= ? where next_val=?
14:35:51,097 DEBUG [org.hibernate.SQL] - insert into Author (firstName, lastName, version, id) values (?, ?, ?, ?)

你可以通过定义一个 @GenericGenerator 来避免这一点,以下代码告诉 Hibernate 使用数据库本地策略生成主键值。

@Entity
public class Author {
 
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO, generator = "native")
    @GenericGenerator(name = "native", strategy = "native")
    @Column(name = "id", updatable = false, nullable = false)
    private Long id;
 
    ...
}

然后,Hibernate 将使用 MySQL 自增数据库列来生成主键值。

14:41:34,255 DEBUG [org.hibernate.SQL] - insert into Author (firstName, lastName, version) values (?, ?, ?)
14:41:34,298 DEBUG [org.hibernate.id.IdentifierGeneratorHelper] - Natively generated identity: 1

 


版权声明:本文为xiaomisolo原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。