development

Java에서 NoClassDefFoundError가 발생하는 이유는 무엇입니까?

big-blog 2020. 2. 12. 23:39
반응형

Java에서 NoClassDefFoundError가 발생하는 이유는 무엇입니까?


NoClassDefFoundErrorJava 애플리케이션을 실행할 때을 받고 있습니다. 일반적으로 이것의 원인은 무엇입니까?


이것은 코드가 의존하는 클래스 파일이 있고 컴파일 타임에 존재하지만 런타임에는 찾을 수 없을 때 발생합니다. 빌드 시간과 런타임 클래스 경로의 차이점을 찾으십시오.


컴파일 타임과 런타임 사이의 클래스 경로 불일치 때문일 수도 있지만 반드시 그런 것은 아닙니다.

이 경우 두세 가지 예외를 머리에 똑바로 유지하는 것이 중요합니다.

  1. java.lang.ClassNotFoundException이 예외는 클래스가 클래스 경로에서 발견되지 않았 음을 나타냅니다. 이것은 클래스 정의를로드하려고했지만 클래스가 클래스 경로에 존재하지 않았 음을 나타냅니다.

  2. java.lang.NoClassDefFoundError이 예외는 JVM이 클래스 정의에 대한 내부 클래스 정의 데이터 구조를 찾아서 찾지 못했음을 나타냅니다. 이것은 클래스 경로에서로드 할 수 없다는 것과 다릅니다. 일반적으로 이것은 이전에 클래스 경로에서 클래스를로드하려고 시도했지만 어떤 이유로 실패했습니다. 이제 클래스를 다시 사용하려고 시도하므로 마지막으로 실패했기 때문에로드해야합니다. 이전에로드하지 못했기 때문에 다시로드하지 않을 것입니다. 이전 실패는 ClassNotFoundException 또는 ExceptionInInitializerError (정적 초기화 블록의 실패를 표시 함) 또는 기타 여러 가지 문제점 일 수 있습니다. 요점은 NoClassDefFoundError가 반드시 클래스 경로 문제 일 필요는 없다는 것입니다.


다음은 설명하는 코드입니다 java.lang.NoClassDefFoundError. 자세한 설명 Jared의 답변참조하십시오 .

NoClassDefFoundErrorDemo.java

public class NoClassDefFoundErrorDemo {
    public static void main(String[] args) {
        try {
            // The following line would throw ExceptionInInitializerError
            SimpleCalculator calculator1 = new SimpleCalculator();
        } catch (Throwable t) {
            System.out.println(t);
        }
        // The following line would cause NoClassDefFoundError
        SimpleCalculator calculator2 = new SimpleCalculator();
    }

}

SimpleCalculator.java

public class SimpleCalculator {
    static int undefined = 1 / 0;
}

Java에서 NoClassDefFoundError

정의:

  1. Java Virtual Machine은 컴파일 타임에 사용 가능한 런타임에 특정 클래스를 찾을 수 없습니다.

  2. 컴파일 시간 동안 클래스가 존재하지만 런타임 동안 Java 클래스 경로에서 사용할 수없는 경우

여기에 이미지 설명을 입력하십시오

예 :

  1. 클래스가 클래스 경로에 있지 않습니다. 알 수있는 확실한 방법은 없지만 System.getproperty ( "java.classpath")를 인쇄하여 최소한 클래스 경로를 인쇄합니다. 실제 런타임 클래스 경로에 대한 아이디어.
  2. NoClassDefFoundError의 간단한 예는 클래스가 누락 된 JAR 파일에 속하거나 JAR이 클래스 경로에 추가되지 않았거나 때로는 내 동료 중 하나가 tibco.jar을 tibco_v3.jar로 변경하고 프로그램이 java.lang.NoClassDefFoundError로 실패하고 무엇이 잘못되었는지 궁금합니다.

  3. 작동 할 것이라고 생각하는 클래스 경로와 함께 명시 적으로 -classpath 옵션으로 실행하려고 시도하고 작동하면 누군가가 Java 클래스 경로를 재정의한다는 짧은 신호입니다.

  4. JAR 파일에 대한 권한 문제로 인해 Java에서 NoClassDefFoundError가 발생할 수도 있습니다.
  5. XML 구성의 오타도 Java에서 NoClassDefFoundError를 일으킬 수 있습니다.
  6. 패키지에 정의 된 컴파일 된 클래스가 JApplet의 경우와 같이로드하는 동안 동일한 패키지에 존재하지 않으면 Java에서 NoClassDefFoundError가 발생합니다.

가능한 해결책:

  1. 이 클래스는 Java 클래스 경로에서 사용할 수 없습니다.
  2. J2EE 환경에서 여러 클래스 로더 중 클래스의 가시성보다 java.lang.NoClassDefFoundError가 발생할 수있는 경우보다 자세한 내용은 예제 및 시나리오 섹션을 참조하십시오.
  3. 로그 파일에서 java.lang.ExceptionInInitializerError를 확인하십시오. 정적 초기화 실패로 인한 NoClassDefFoundError는 매우 일반적입니다.
  4. NoClassDefFoundError는 java.lang.LinkageError의 서브 클래스이므로 네이티브 라이브러리와 같은 종속성 중 하나를 사용할 수없는 경우에도 발생할 수 있습니다.
  5. 모든 시작 스크립트가 Classpath 환경 변수를 대체합니다.
  6. jar 명령을 사용하여 프로그램을 실행 중일 수 있으며 매니페스트 파일의 ClassPath 속성에 클래스가 정의되어 있지 않습니다.

자원:

NoClassDefFoundError를 해결하는 3 가지 방법

java.lang.NoClassDefFoundError 문제 패턴


코드가 런타임에 발견 된 호환되지 않는 클래스의 클래스로 컴파일 될 때 때때로 NoClassDefFound 오류가 발생한다는 것을 알았습니다. 내가 기억하는 특정 인스턴스는 아파치 축 라이브러리입니다. 실제로 런타임 클래스 경로에 2 가지 버전이 있었고 오래된 버전과 호환되지 않는 버전을 선택했지만 올바른 버전이 아니므로 NoClassDefFound 오류가 발생했습니다. 이것은 비슷한 명령을 사용하는 명령 줄 앱에있었습니다.

set classpath=%classpath%;axis.jar

다음을 사용하여 올바른 버전을 선택할 수있었습니다.

set classpath=axis.jar;%classpath%;

이것이 지금까지 찾은 최고의 솔루션 입니다.

org.mypackage클래스를 포함 하는 패키지가 있다고 가정 합니다.

  • HelloWorld (메인 클래스)
  • 지원 클래스
  • UtilClass

이 패키지를 정의하는 파일은 디렉토리 D:\myprogram(Windows) 또는 /home/user/myprogram(Linux) 에 실제로 저장됩니다 .

파일 구조는 다음과 같습니다. 여기에 이미지 설명을 입력하십시오

Java를 호출 할 때 실행할 애플리케이션 이름을 지정합니다 org.mypackage.HelloWorld. 그러나 패키지를 정의하는 파일과 디렉토리를 찾을 위치를 Java에 알려야합니다. 따라서 프로그램을 시작하려면 다음 명령을 사용해야합니다.여기에 이미지 설명을 입력하십시오


Maven함께 Spring Framework사용 하고 있었고 프로젝트 에서이 오류를 해결했습니다.

클래스에 런타임 오류가있었습니다. 속성을 정수로 읽는 중이지만 속성 파일에서 값을 읽을 때 값이 두 배였습니다.

Spring은 런타임이 실패한 라인에 대한 전체 스택 추적을 제공하지 않았습니다. 단순히 말했다 NoClassDefFoundError. 그러나 네이티브 Java 응용 프로그램으로 실행하면 (MVC에서 가져옴) 이것이 ExceptionInInitializerError진정한 원인이었고 오류를 추적 한 방법입니다.

@xli의 대답은 내 코드에서 무엇이 잘못되었는지에 대한 통찰력을 제공했습니다.


런타임 클래스 로더가로드 한 클래스가 Java 루트 로더가 이미로드 한 클래스에 액세스 할 수없는 경우 NoClassFoundError가 발생합니다. java에 따라 다른 클래스 로더가 다른 보안 도메인에 있기 때문에 jvm은 루트 로더에 의해 이미로드 된 클래스가 런타임 로더 주소 공간에서 분석되는 것을 허용하지 않습니다.

'java -javaagent : tracer.jar [YOUR java ARGS]'로 프로그램을 실행하십시오.

로드 된 클래스와 클래스를로드 한 로더 env를 표시하는 출력을 생성합니다. 클래스를 확인할 수없는 이유를 추적하면 매우 유용합니다.

// ClassLoaderTracer.java
// From: https://blogs.oracle.com/sundararajan/entry/tracing_class_loading_1_5

import java.lang.instrument.*;
import java.security.*;

// manifest.mf
// Premain-Class: ClassLoadTracer

// jar -cvfm tracer.jar manifest.mf ClassLoaderTracer.class

// java -javaagent:tracer.jar  [...]

public class ClassLoadTracer 
{
    public static void premain(String agentArgs, Instrumentation inst) 
    {
        final java.io.PrintStream out = System.out;
        inst.addTransformer(new ClassFileTransformer() {
            public byte[] transform(ClassLoader loader, String className, Class classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException {

                String pd = (null == protectionDomain) ? "null" : protectionDomain.getCodeSource().toString();
                out.println(className + " loaded by " + loader + " at " + new java.util.Date() + " in " + pd);

                // dump stack trace of the thread loading class 
                Thread.dumpStack();

                // we just want the original .class bytes to be loaded!
                // we are not instrumenting it...
                return null;
            }
        });
    }
}

많은 것을 볼 수있는 흥미로운 사례는 NoClassDefFoundErrors다음과 같습니다.

  1. throwRuntimeException에서 static클래스의 블록Example
  2. 그것을 가로 채십시오 (또는 테스트 케이스 에서 던져지는 것처럼 중요하지 않은 경우 )
  3. 이 클래스의 인스턴스를 만들어보십시오 Example

static class Example {
    static {
        thisThrowsRuntimeException();
    }
}

static class OuterClazz {

    OuterClazz() {
        try {
            new Example();
        } catch (Throwable ignored) { //simulating catching RuntimeException from static block
            // DO NOT DO THIS IN PRODUCTION CODE, THIS IS JUST AN EXAMPLE in StackOverflow
        }

        new Example(); //this throws NoClassDefFoundError
    }
}

NoClassDefErrorExceptionInInitializerError정적 블록 과 함께 발생 RuntimeException합니다.


UNIT TESTSNoClassDefFoundErrors 에서 볼 때 특히 중요 합니다 .

어떤 방식으로 static테스트간에 블록 실행을 "공유" 하지만 초기 ExceptionInInitializerError는 하나의 테스트 사례에 불과합니다. 문제가있는 Example클래스 를 사용하는 첫 번째 클래스입니다. Example클래스 를 사용하는 다른 테스트 케이스 는 그냥 던져 질 것 NoClassDefFoundErrors입니다.


생성 된 코드 (EMF 등)가있는 경우 모든 스택 공간을 사용하는 정적 이니셜 라이저가 너무 많을 수 있습니다.

스택 오버플로 질문 Java 스택 크기를 늘리는 방법을 참조하십시오 . .


아래 기술은 여러 번 도움이되었습니다.

System.out.println(TheNoDefFoundClass.class.getProtectionDomain().getCodeSource().getLocation());

여기서 TheNoDefFoundClass는 프로그램에서 사용하는 동일한 라이브러리의 이전 버전에 대한 환경 설정으로 인해 "손실 된"클래스입니다. 이는 클라이언트 소프트웨어가 자체 클래스 로더와 수많은 고대 버전의 가장 인기있는 라이브러리로 구성된 지배적 인 컨테이너에 배포되는 경우에 가장 자주 발생합니다.


모든 모듈에 대해 preDexLibraries를 비활성화하여 문제를 해결했습니다.

dexOptions {
        preDexLibraries false
        ...

NoClassDefFoundError또한 정적 초기화 프로그램이 런타임에서 사용할 수없는 자원 번들 (예 : 영향을받는 클래스가 META-INF디렉토리 에서로드하려고 시도 하지만 존재하지 않는 특성 파일)을로드하려고 할 때 발생할 수 있습니다 . 잡지 않으면 NoClassDefFoundError때로는 전체 스택 추적을 볼 수 없습니다. 이를 극복하기 위해 일시적으로 다음 catch절을 사용할 수 있습니다 Throwable.

try {
    // Statement(s) that cause the affected class to be loaded
} catch (Throwable t) {
    Logger.getLogger("<logger-name>").info("Loading my class went wrong", t);
}

누군가가 java.lang.NoClassDefFoundError: org/apache/log4j/Logger오류 때문에 여기에 오면 , 내 경우에는 log4j 2를 사용했기 때문에 생성되었지만 (제공되는 모든 파일을 추가하지는 않았습니다) 일부 종속성 라이브러리는 log4j 1을 사용했습니다. 해결책은 Log4j를 추가하는 것이 었습니다 1.x 브리지 : log4j-1.2-api-<version>.jarlog4j와 함께 제공 되는 jar 2. log4j 2 마이그레이션 에 대한 추가 정보 .


동일한 프로젝트의 서로 다른 두 개의 체크 아웃 사본

필자의 경우 문제는 Eclipse가 동일한 프로젝트의 서로 다른 두 사본을 구별 할 수 없다는 점이었습니다. 하나는 트렁크 (SVN 버전 제어)에 잠겨 있고 다른 하나는 한 번에 한 지점에서 작동합니다. JUnit 테스트 사례로 작업 복사본의 한 가지 변경 사항을 시도했습니다. 여기서는 개인 내부 클래스를 자체적으로 공개 클래스로 추출하는 작업이 포함되었으며 작업하는 동안 프로젝트의 다른 복사본을 열어 다른 것을 둘러 봅니다. 변경이 필요한 코드의 일부. 어느 시점에서, NoClassDefFoundError개인 내부 계급이 존재하지 않았다고 불평하는 팝업 나타났습니다. 스택 추적을 두 번 클릭하면 잘못된 프로젝트 복사본의 소스 파일로 이동했습니다.

프로젝트의 트렁크 사본을 닫고 테스트 사례를 다시 실행하면 문제가 해결되었습니다.


이 오류는 확인되지 않은 Java 버전 요구 사항 으로 인해 발생할 수 있습니다 .

필자의 경우 SDKMAN을 사용하여 Java 9에서 Java 8로 전환하여 고급 오픈 소스 프로젝트를 구축하는 동안이 오류를 해결할 수있었습니다 ! .

sdk list java
sdk install java 8u152-zulu
sdk use java 8u152-zulu

그런 다음 아래 설명에 따라 새로 설치하십시오.


사용하는 경우 메이븐을 할, 일반적으로 떠오르고 - 빌드 도구로, 때로는 도움이 깨끗한 '설치'빌드를 사용하지 테스트와 함께 .

mvn clean install -DskipTests

이제 모든 내장 설치되어, 당신은 가서 테스트를 실행할 수 있습니다.

mvn test

프로젝트의 Java 빌드 경로에있는 "주문 및 내보내기"탭에서 클래스를 내 보내지 않은 경우 NoClassDefFound 오류가 발생했습니다. 프로젝트 빌드 경로에 추가 한 모든 종속성의 "주문 및 내보내기"탭에 확인 표시를하십시오. Eclipse 경고 : XXXXXXXXXXX.jar을 내보내거나 공개하지 않습니다를 참조하십시오 . 런타임 ClassNotFoundExceptions가 발생할 수 있습니다 .


특정 패키지 이름의 IDE에서 코드 파일을 복사하고 터미널을 사용하여 코드 파일을 실행하려고 할 수도 있습니다. 코드에서 패키지 이름을 먼저 제거해야합니다. 이것은 나에게 일어난다.


Java가 런타임에서 클래스 A를 찾을 수 없습니다. 클래스 A는 다른 작업 공간에서 Maven 프로젝트 ArtClient에있었습니다. 그래서 ArtClient를 Eclipse 프로젝트로 가져 왔습니다. 내 프로젝트 중 두 개가 ArtClient를 종속성으로 사용하고있었습니다. 이 라이브러리 참조를 프로젝트 참조로 변경했습니다 (빌드 경로-> 빌드 경로 구성).

그리고 문제는 사라졌습니다.


나는 같은 문제를 겪었고 몇 시간 동안 주식을 가지고있었습니다.

해결책을 찾았습니다. 제 경우에는 그로 인해 정적 메서드가 정의되었습니다. JVM이 해당 클래스의 다른 오브젝트를 작성할 수 없습니다.

예를 들어

private static HttpHost proxy = new HttpHost(proxyHost, Integer.valueOf(proxyPort), "http");

SRC 라이브러리에서 두 개의 파일을 제거한 후이 메시지가 나타 났으며 다시 가져 왔을 때이 오류 메시지가 계속 나타납니다.

내 해결책은 : Eclipse를 다시 시작하십시오. 그 이후로이 메시지를 다시 보지 못했습니다 :-)


module:app에서 일치하는지 확인하십시오 module:lib.

android {
    compileSdkVersion 23
    buildToolsVersion '22.0.1'
    packagingOptions {
    }

    defaultConfig {
        minSdkVersion 17
        targetSdkVersion 23
        versionCode 11
        versionName "2.1"
    }

참고 URL : https://stackoverflow.com/questions/34413/why-am-i-getting-a-noclassdeffounderror-in-java



반응형