2

在我们的应用程序中,我是另一个系统的客人,我不想做出任何具有大规模后果的改变。

我正在从头开始编写单元测试,因为之前的作者没有打扰。(抱怨。)我正在尝试插入一行以使用 JPA/hibernate 测试我的查询,但很惊讶我收到以下错误:

javax.persistence.PersistenceException: org.hibernate.exception.ConstraintViolationException: integrity constraint violation: NOT NULL check constraint; SYS_CT_14316 table: OPTIONVALUE column: OPTION_INDEX

    at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1377)
    at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1300)
    at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1306)
    at org.hibernate.ejb.AbstractEntityManagerImpl.flush(AbstractEntityManagerImpl.java:989)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler.invoke(SharedEntityManagerCreator.java:293)
    at com.sun.proxy.$Proxy109.flush(Unknown Source)

尤其是当我知道我已经设置了值时,因为我必须为它编写一个 setter。然后,在其中一个对象上,我找到了以下定义(我必须查找):

public static final String OPTION_VALUE_POSITION_COLUMN = "option_index";
@Column(name = OPTION_VALUE_POSITION_COLUMN, insertable = false, updatable = false, nullable = false)
private int position;

这就解释了为什么我在该列上仍然为空。

我在删除 insertable 时没有什么大问题,因为它不是自动生成的,所以如果你要插入一条记录,为什么你不想包含它,这显然不会在系统内部发生。并且可更新将保护它。但如果我不需要,我真的不想。

所以,我的问题是,有没有办法只为单元测试覆盖它。我尝试在谷歌上搜索几件事,但为此措辞有点问题,我什么也没找到。

4

2 回答 2

1

通常你可以重新定义测试包中的任何类,它会被加载而不是主类,因为它具有更高的优先级。类加载路径通常按以下顺序排列优先级:

  1. 测试类(如果运行测试)
  2. 测试资源(如果正在运行测试)
  3. 主要课程
  4. 主要资源
  5. 测试依赖 jars(如果正在运行测试)
  6. 主要依赖jar

您可以像这样检查它:

// I have yet to run into a custom ClassLoader that does not extend URLClassLoader
URLClassLoader urls = (URLClassLoader) getClass().getClassLoader();
for (URL url : urls.getURLs()) {
    System.out.println(url);
}

不过,这里有点问题。Hibernate 的 ClassLoader 可能(也可能不会)从 Test 中读取被覆盖的类,除非你有你的配置文件persistence.xmlorm.xml或者在 Test Resources 中的任何东西。这可能是也可能不是,但是如果你需要在测试中有一个单独的配置,那么它必须列出所有需要扫描的类。您不能依赖该<exclude-unlisted-classes>false</exclude-unlisted-classes>条目,因为 Hibernate 的扫描器是深度感知的,并且在 Class-Path 层次结构中不会比它找到配置 XML 的级别更深。

于 2018-08-01T09:37:25.793 回答
0

我不知道这是否是最好的解决方案,但在某些情况下它可能是一个合法的解决方法。

在这些情况下,我使用原始查询插入/更新我的实体:

Query query = entityManager.createNativeQuery("insert into MY_TABLE " + 
            "(ID, NAME) values (1, 'user1')");
query.executeUpdate();
于 2021-04-06T13:26:58.193 回答