NoClassDefFoundError错误码

10

我正在尝试在我的Spring Boot应用程序中运行JUnit测试,但是出现了下面所示的NoClassDefFoundError错误。

pom.xml文件中同时包含spring-boot-starter-parent和spring-boot-starter-test(v2.0.5.RELEASE)。我也添加了spring-core(v5.0.9.RELEASE)。

显然,ErrorCoded类自4.3.6起已被弃用,因此我不确定如何找出测试运行程序(或其他库)仍在尝试加载此类的原因。

这是我的测试类:

@RunWith(SpringJUnit4ClassRunner.class)
public class SystemBuilderTest {
    
    private System system;

    @Before
    public void setUp() throws Exception {
        StatefulConnection conn = new StatefulConnection.Builder(null)
                .build();
        
        Device d1 = new SensingDevice.Builder("sensor1", conn)
                .build();
        
        system = new System.SystemBuilder("testSystem")
                .addChildDevice(d1)
                .build();
        
        system.initialize();
    }

    @Test
    public void testStart() throws DCFDeviceException {
        system.start();
        assertTrue(system.getName().equals("testSystem"));
        assertTrue(system.getChildDevices().size() == 1);
        assertTrue(system.getChildDevices().iterator().next().getName().equals("sensor1"));
        
    }
} 

...和堆栈跟踪:

java.lang.NoClassDefFoundError: org/springframework/core/ErrorCoded
    at java.base/java.lang.ClassLoader.defineClass1(Native Method)
    at java.base/java.lang.ClassLoader.defineClass(ClassLoader.java:1007)
    at java.base/java.security.SecureClassLoader.defineClass(SecureClassLoader.java:174)
    at java.base/jdk.internal.loader.BuiltinClassLoader.defineClass(BuiltinClassLoader.java:801)
    at java.base/jdk.internal.loader.BuiltinClassLoader.findClassOnClassPathOrNull(BuiltinClassLoader.java:699)
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClassOrNull(BuiltinClassLoader.java:622)
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:580)
    at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:185)
    at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:496)
    at java.base/java.lang.ClassLoader.defineClass1(Native Method)
    at java.base/java.lang.ClassLoader.defineClass(ClassLoader.java:1007)
    at java.base/java.security.SecureClassLoader.defineClass(SecureClassLoader.java:174)
    at java.base/jdk.internal.loader.BuiltinClassLoader.defineClass(BuiltinClassLoader.java:801)
    at java.base/jdk.internal.loader.BuiltinClassLoader.findClassOnClassPathOrNull(BuiltinClassLoader.java:699)
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClassOrNull(BuiltinClassLoader.java:622)
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:580)
    at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:185)
    at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:496)
    at org.springframework.context.support.GenericApplicationContext.<init>(GenericApplicationContext.java:110)
    at org.springframework.test.context.support.AbstractGenericContextLoader.loadContext(AbstractGenericContextLoader.java:115)
    at org.springframework.test.context.support.AbstractGenericContextLoader.loadContext(AbstractGenericContextLoader.java:60)
    at org.springframework.test.context.support.AbstractDelegatingSmartContextLoader.delegateLoading(AbstractDelegatingSmartContextLoader.java:275)
    at org.springframework.test.context.support.AbstractDelegatingSmartContextLoader.loadContext(AbstractDelegatingSmartContextLoader.java:251)
    at org.springframework.test.context.cache.DefaultCacheAwareContextLoaderDelegate.loadContextInternal(DefaultCacheAwareContextLoaderDelegate.java:99)
    at org.springframework.test.context.cache.DefaultCacheAwareContextLoaderDelegate.loadContext(DefaultCacheAwareContextLoaderDelegate.java:117)
    at org.springframework.test.context.support.DefaultTestContext.getApplicationContext(DefaultTestContext.java:108)
    at org.springframework.test.context.support.DependencyInjectionTestExecutionListener.injectDependencies(DependencyInjectionTestExecutionListener.java:117)
    at org.springframework.test.context.support.DependencyInjectionTestExecutionListener.prepareTestInstance(DependencyInjectionTestExecutionListener.java:83)
    at org.springframework.test.context.TestContextManager.prepareTestInstance(TestContextManager.java:246)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.createTest(SpringJUnit4ClassRunner.java:227)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner$1.runReflectiveCall(SpringJUnit4ClassRunner.java:289)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.methodBlock(SpringJUnit4ClassRunner.java:291)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:246)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:97)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
    at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:190)
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:86)
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:538)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:760)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:460)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:206)
Caused by: java.lang.ClassNotFoundException: org.springframework.core.ErrorCoded
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:582)
    at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:185)
    at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:496)
    ... 50 common frames omitted

你能从pom中移除spring-core(v5.0.9.RELEASE)并检查吗? - kj007
我从pom中删除了spring-core,但错误仍然出现。 - Eric Hansen
请问您能否添加您的Pom文件? - kj007
4个回答

7

你的构建中有一些东西依赖于spring-test 4.3.x(不确定是哪个小版本,但这并不重要)。

我之所以这么说是因为堆栈跟踪中的行号与5.x版本中的SpringJUnit4ClassRunner类中的行号不对齐。

从你的堆栈跟踪中可以看到:

at o.s.t.c.junit4.SpringJUnit4ClassRunner.createTest(SpringJUnit4ClassRunner.java:227)

这说明在第227行应该调用一个名为createTest的方法。您可以在此类的4.3.x分支版本中看到这是正确的。
但在5.0.x分支版本中,第227行改为调用getTestContextManager().prepareTestInstance(testInstance);
这非常明确。因此,如果您的IDE告诉您相反(如您在另一条评论中所指出的),则是错误的。或者如果IDE支持该功能,则可能需要在项目上进行“清理”。
由于您没有发布完整的POM文件,因此无法告诉您是什么原因导致旧依赖项被引入。但是,您可以使用Maven依赖插件来找出原因。
mvn dependency:tree

这个命令将以树形式打印依赖关系。建议将输出重定向到文件并使用grep查找spring-test。这样可以识别出有问题的依赖项,并使用maven <exclusion>禁用它。


spring-statemachine和flowable引入了较旧的版本...所以我在这些依赖项中排除了spring-context。 - Eric Hansen

0

你能否使用SpringRunner.class@SpringBootTest来运行你的测试,以便加载所有必需的测试依赖项。

@RunWith(SpringRunner.class)
@SpringBootTest
public class SystemBuilderTest {

SpringRunnerSpringJUnit4ClassRunner的别名,因此更改没有任何好处。 - Roddy of the Frozen Peas

0

我无法准确回答这个错误的原因,但我可以告诉你,你的某个依赖版本存在冲突。

正如下面的文档所示(尽管您使用的是版本2.0.5.RELEASE)。

https://docs.spring.io/spring-boot/docs/2.0.4.RELEASE/reference/htmlsingle/

ErrorCoded被SpringJUnit4ClassRunner使用,但如果你在他们的Github上看到,SpringJUnit4ClassRunner在org.springframework.spring-test的5.0.8版本中不再使用ErrorCoded。

你必须查看你的Spring项目实际使用的是哪个版本的org.springframework.spring-test,因为SpringJUnit4ClassRunner在该jar包中。

希望这可以帮助你。


已解决的依赖项列表(Eclipse Maven插件)显示spring-test为5.0.9.RELEASE。 - Eric Hansen
是的,那是因为你使用的是Spring版本2.0.5.RELEASE。我的意思是,spring-test的版本5.0.8根本没有使用ErrorCoded,5.0.9也是如此(这在它们的GIthub上有证据)。你尝试重新创建你的项目了吗? - trjade

0

对我来说,spring-security-test 拉取了一个旧版本的 spring-test。我正在使用 spring-boot 2.x

λ mvn dependency:tree | grep test
[INFO] +- org.springframework.security:spring-security-test:jar:4.2.3.RELEASE:test
[INFO] |  \- org.springframework:spring-test:jar:4.3.9.RELEASE:test

将spring-security和spring-security-test更新到5.x版本即可解决问题。

<properties>
    <java.version>1.8</java.version>
    <spring.version>2.0.4.RELEASE</spring.version>
    <logging.version>2.8.2</logging.version>
    <spring-security-test.version>5.0.6.RELEASE</spring-security-test.version>
    <spring-security-config.version>5.0.6.RELEASE</spring-security-config.version>
</properties>

<dependencies>
    <dependency>
        <groupId>org.springframework.security</groupId>
        <artifactId>spring-security-core</artifactId>
        <version>${spring-security-config.version}</version>
        <scope>compile</scope>
    </dependency>

    <dependency>
        <groupId>org.springframework.security</groupId>
        <artifactId>spring-security-web</artifactId>
        <version>${spring-security-config.version}</version>
        <scope>compile</scope>
    </dependency>

    <dependency>
        <groupId>org.springframework.security</groupId>
        <artifactId>spring-security-test</artifactId>
        <version>${spring-security-test.version}</version>
        <scope>test</scope>
    </dependency>
</dependencies>

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接