SequenceGenerator 在 JUnit 测试中不起作用?
SequenceGenerator does not work in JUnit test?
我尝试使用内存中的 H2 DB 测试某些实体的持久性,但我认识到 @SequenceGenerator
永远不会被调用,构建平台也不会调用 运行, 运行 在 Eclipse 中使用 RunAs->JUnit 测试时也不会。
我可以肯定地说,这些序列是在 H2 数据库中生成的。当我连接到这个生成的 H2 时,我什至可以 select 它们。所以这绝对不是 H2 内部的问题,而是 Hibernate 的问题。
(通常 Hibernate 在持久化需要一个 ID 的实体时会自动分配一个 ID)。
实体
@Entity
@Table(name = "HOUSE_USERDATA")
public class UserData {
@Id
@Column(name = "HU_ID")
@GeneratedValue(generator = "SEQ_HOUSE_USERDATA", strategy = GenerationType.SEQUENCE)
@SequenceGenerator(sequenceName = "SEQ_HOUSE_USERDATA", name = "SEQ_HOUSE_USERDATA", allocationSize = 2)
private Long huId;
@Column(name = "HU_DATA")
@Size(max = 1000)
private String m_data;
@ManyToOne
@JoinColumn(name = "HR_ID")
private Registry m_registry;
//more code [...]
}
引用实体中的引用...
@OneToMany(mappedBy = "registry")
private List<UserData> userDataList;
持久化单元...
<persistence-unit name="test" transaction-type="RESOURCE_LOCAL">
<provider>org.hibernate.ejb.HibernatePersistence</provider>
<class>com.foo.bar.all.entity</class>
<!-- all entity references -->
<exclude-unlisted-classes>false</exclude-unlisted-classes>
<properties>
<property name="hibernate.archive.autodetection" value="class"/>
<property name="hibernate.connection.username" value="sa"/>
<property name="hibernate.connection.password" value=""/>
<property name="hibernate.connection.driver_class" value="org.h2.Driver"/>
<property name="hibernate.connection.url"
value="jdbc:h2:inmemory;INIT=runscript from 'classpath:testscripts/drop_h2.sql'\;runscript from 'classpath:testscripts/create.sql'"/>
<property name="hibernate.dialect" value="org.hibernate.dialect.H2Dialect" />
<property name="hibernate.hbm2ddl.auto" value="create-drop"/>
<property name="hibernate.show_sql" value="true" />
<property name="hibernate.format_sql" value="true" />
<property name="hibernate.use_sql_comments" value="true" />
</properties>
</persistence-unit>
JUnit测试中的调用...
Registry registry = new Registry();
registry.setClientId("clientId");
List<UserData> userDataList = new ArrayList<>();
UserData userData1 = new UserData();
userData1.setData("User defined data 1.");
userData1.setRegistry(registry);
UserData userData2 = new UserData();
userData2.setData("User defined data 2.");
userData2.setRegistry(registry);
userDataList.add(userData1);
userDataList.add(userData2);
registry.setUserDataList(userDataList);
entityManager.persist(registry);
Registry result = entityManager.find(Registry.class, "clientId");
//MUST NOT BE NULL, BUT IS NULL
assertThat(result.getUserDataList().get(0).getId(), is(not(nullValue())))
其他值已正确保留。只有 ID 没有生成。 (我想知道为什么这个测试对所有其他值都有效,因为 ID 在生成的数据库中被定义为 NOT NULL,所以应该有一个持久性异常或其他东西)。
知道为什么序列生成器不生成任何东西吗(我也试过 GenerationType.AUTO
,但没有区别)?
当您正在执行 entityManager.persist(registry)
时,存储 Registry
并检查 class 的所有映射。它会遇到 UserData
个对象的集合,但是因为没有 cascade
属性 匹配 PERSIST
它不会存储 UserData
个对象。
它只会存储顶级 Registry
对象。如果您想更改此添加 cascade={CascadeType.ALL}
或至少 cascade={CascadeType.PERSIST}
到 @OneToMany
注释,以告诉 Hibernate 它还需要检查集合中的新元素并持久化这些元素。
或者先存储UserData
个元素,再存储Registry
个元素。
我尝试使用内存中的 H2 DB 测试某些实体的持久性,但我认识到 @SequenceGenerator
永远不会被调用,构建平台也不会调用 运行, 运行 在 Eclipse 中使用 RunAs->JUnit 测试时也不会。
我可以肯定地说,这些序列是在 H2 数据库中生成的。当我连接到这个生成的 H2 时,我什至可以 select 它们。所以这绝对不是 H2 内部的问题,而是 Hibernate 的问题。 (通常 Hibernate 在持久化需要一个 ID 的实体时会自动分配一个 ID)。
实体
@Entity
@Table(name = "HOUSE_USERDATA")
public class UserData {
@Id
@Column(name = "HU_ID")
@GeneratedValue(generator = "SEQ_HOUSE_USERDATA", strategy = GenerationType.SEQUENCE)
@SequenceGenerator(sequenceName = "SEQ_HOUSE_USERDATA", name = "SEQ_HOUSE_USERDATA", allocationSize = 2)
private Long huId;
@Column(name = "HU_DATA")
@Size(max = 1000)
private String m_data;
@ManyToOne
@JoinColumn(name = "HR_ID")
private Registry m_registry;
//more code [...]
}
引用实体中的引用...
@OneToMany(mappedBy = "registry")
private List<UserData> userDataList;
持久化单元...
<persistence-unit name="test" transaction-type="RESOURCE_LOCAL">
<provider>org.hibernate.ejb.HibernatePersistence</provider>
<class>com.foo.bar.all.entity</class>
<!-- all entity references -->
<exclude-unlisted-classes>false</exclude-unlisted-classes>
<properties>
<property name="hibernate.archive.autodetection" value="class"/>
<property name="hibernate.connection.username" value="sa"/>
<property name="hibernate.connection.password" value=""/>
<property name="hibernate.connection.driver_class" value="org.h2.Driver"/>
<property name="hibernate.connection.url"
value="jdbc:h2:inmemory;INIT=runscript from 'classpath:testscripts/drop_h2.sql'\;runscript from 'classpath:testscripts/create.sql'"/>
<property name="hibernate.dialect" value="org.hibernate.dialect.H2Dialect" />
<property name="hibernate.hbm2ddl.auto" value="create-drop"/>
<property name="hibernate.show_sql" value="true" />
<property name="hibernate.format_sql" value="true" />
<property name="hibernate.use_sql_comments" value="true" />
</properties>
</persistence-unit>
JUnit测试中的调用...
Registry registry = new Registry();
registry.setClientId("clientId");
List<UserData> userDataList = new ArrayList<>();
UserData userData1 = new UserData();
userData1.setData("User defined data 1.");
userData1.setRegistry(registry);
UserData userData2 = new UserData();
userData2.setData("User defined data 2.");
userData2.setRegistry(registry);
userDataList.add(userData1);
userDataList.add(userData2);
registry.setUserDataList(userDataList);
entityManager.persist(registry);
Registry result = entityManager.find(Registry.class, "clientId");
//MUST NOT BE NULL, BUT IS NULL
assertThat(result.getUserDataList().get(0).getId(), is(not(nullValue())))
其他值已正确保留。只有 ID 没有生成。 (我想知道为什么这个测试对所有其他值都有效,因为 ID 在生成的数据库中被定义为 NOT NULL,所以应该有一个持久性异常或其他东西)。
知道为什么序列生成器不生成任何东西吗(我也试过 GenerationType.AUTO
,但没有区别)?
当您正在执行 entityManager.persist(registry)
时,存储 Registry
并检查 class 的所有映射。它会遇到 UserData
个对象的集合,但是因为没有 cascade
属性 匹配 PERSIST
它不会存储 UserData
个对象。
它只会存储顶级 Registry
对象。如果您想更改此添加 cascade={CascadeType.ALL}
或至少 cascade={CascadeType.PERSIST}
到 @OneToMany
注释,以告诉 Hibernate 它还需要检查集合中的新元素并持久化这些元素。
或者先存储UserData
个元素,再存储Registry
个元素。