찾다
Javajava지도 시간Java 동적 클래스 로딩 및 재로딩에 대한 자세한 소개

이 기사는 Java 동적 클래스 로딩 및 재로딩에 대한 자세한 소개를 제공합니다. 도움이 필요한 친구들이 참고할 수 있기를 바랍니다.

클래스는 Java에서 런타임에 로드하고 다시 로드할 수 있지만 생각만큼 간단하지는 않습니다. 이 기사에서는 Java에서 클래스를 로드하고 다시 로드하는 시기와 방법을 설명합니다.
동적으로 클래스를 로드하는 것이 Java 리플렉션의 일부인지 Java 코어의 일부인지 논쟁할 수 있습니다. 어쨌든 더 좋은 곳이 없어서 Java Reflection에 넣었습니다.

클래스 로더

Java 프로그램의 모든 클래스는 java.lang.ClassLoader의 일부 하위 클래스를 사용하여 로드됩니다. 따라서 동적으로 로드된 클래스는 java.lang.ClassLoader의 하위 클래스도 사용해야 합니다.
클래스가 로드되면 참조하는 클래스도 로드됩니다. 클래스 로딩 모드는 필요한 모든 클래스가 로드될 때까지 반복적으로 로드됩니다. 이는 응용 프로그램의 모든 클래스가 아닐 수도 있습니다. 참조되지 않은 클래스는 참조될 때까지 로드되지 않습니다.

클래스 로딩 계층

클래스 로딩은 Java에서 계층으로 구성됩니다. 독립형 ClassLoader를 생성할 때 상위 ClassLoader를 제공해야 합니다. ClassLoader가 클래스를 로드하라는 요청을 받으면 상위 ClassLoader에 클래스를 로드하도록 요청합니다. 상위 클래스 로더가 클래스를 찾을 수 없으면 하위 클래스 로더는 클래스를 자체적으로 로드하려고 시도합니다.

클래스 로딩

클래스 로더가 클래스를 로드하는 단계는 다음과 같습니다.

  1. 클래스가 로드되었는지 확인하세요

  2. 클래스가 로드되지 않은 경우 상위 클래스 로더에 로드하도록 요청하세요.

  3. 부모 클래스가 로드된 경우 클래스 로더를 로드할 수 없는 경우 현재 클래스 로더를 사용하여 로드해 보세요.

클래스를 오버로드할 수 있는 클래스 로더를 구현할 때 약간의 편차가 필요합니다. 이 시퀀스에서. 다시 로드할 클래스를 로드하도록 부모 클래스 로더에 요청하면 안 됩니다. 이에 대해서는 나중에 자세히 설명하겠습니다.

동적 클래스 로딩

클래스의 동적 로딩은 매우 간단합니다. 여러분이 해야 할 일은 ClassLoader를 가져와서 loadClass() 메서드를 호출하는 것뿐입니다. 예는 다음과 같습니다.

public class MainClass {

  public static void main(String[] args){

    ClassLoader classLoader = MainClass.class.getClassLoader();

    try {
        Class aClass = classLoader.loadClass("com.jenkov.MyClass");
        System.out.println("aClass.getName() = " + aClass.getName());
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }

}

동적 클래스 다시 로드

동적 클래스 다시 로드에는 몇 가지 문제가 있습니다. Java의 내장 클래스 로더는 항상 클래스를 로드하기 전에 로드되었는지 여부를 확인합니다. 따라서 Java의 내장 클래스 로더를 사용하여 클래스를 다시 로드하는 것은 불가능합니다. 클래스를 다시 로드하려면 자신만의 ClassLoader 하위 클래스를 구현해야 합니다.
클래스 로더의 사용자 정의 하위 클래스에도 문제가 있습니다. 로드된 모든 클래스를 연결해야 합니다. 이 메소드는 최종 메소드이므로 ClassLoader 서브클래스에 의해 대체될 수 없습니다. 해결() 메서드는 ClassLoader 인스턴스가 클래스에 두 번 연결되는 것을 허용하지 않습니다. 따라서 클래스를 다시 로드해야 할 때마다 ClassLoader 클래스의 인스턴스를 다시 생성해야 합니다. 불가능한 것은 아니지만 클래스를 다시 로드하도록 언제 디자인해야 하는지 알아야 합니다.

클래스 오버로딩 코드 디자인

위에서 언급했듯이 지정된 클래스를 로드하는 ClassLoader를 사용하여 이 클래스를 다시 로드할 수 없습니다. 따라서 이 클래스를 로드하려면 다른 ClassLoader를 사용해야 합니다. 그러나 이는 새로운 문제를 야기합니다.
Java 프로그램에 로드된 각 클래스는 정규화된 이름(패키지 이름 + 클래스 이름)으로 식별되며 ClassLoader 인스턴스에 의해 로드됩니다. 이는 클래스 로더 A가 로드한 MyObject 클래스가 클래스 로더 B가 로드한 동일한 MyObject 클래스와 다르다는 것을 의미합니다. 시뮬레이션 코드는 다음과 같습니다.

MyObject object = (MyObject)
    myClassReloadingFactory.newInstance("com.jenkov.MyObject");

MyObject 클래스가 코드에서 객체 유형의 변수로 어떻게 참조되는지 확인하세요. 이로 인해 이 클래스의 상주 코드를 이미 로드한 클래스 로더가 MyObject 클래스를 로드합니다.
myClassReloadingFactory 개체 팩토리가 상주 코드와 다른 클래스 로더를 사용하여 MyObject를 로드하는 경우 다시 로드된 개체 유형 변수 MyObject를 MyObject 유형으로 캐스팅할 수 없습니다. 두 개의 MyObject는 서로 다른 클래스 로더에 의해 로드되므로 정규화된 이름이 동일하더라도 서로 다른 클래스로 간주됩니다. 객체의 클래스를 다른 클래스에 대한 참조로 캐스팅하려고 하면 ClassCastException이 발생합니다.
이 제한을 우회하는 것이 가능하지만 두 가지 방법으로 코드를 변경해야 합니다.

  1. 인터페이스를 변수 유형으로 사용하고 구현 클래스만 다시 로드합니다.

  2. 슈퍼클래스를 변수 유형으로 사용하고 하위 클래스만 다시 로드합니다.

샘플 코드는 다음과 같습니다.

MyObjectInterface object = (MyObjectInterface)
    myClassReloadingFactory.newInstance("com.jenkov.MyObject");
MyObjectSuperclass object = (MyObjectSuperclass)
    myClassReloadingFactory.newInstance("com.jenkov.MyObject");

위 코드는 변수 유형이 인터페이스나 슈퍼클래스인 경우 잘 작동합니다. 구현이나 서브클래스를 다시 로드할 때 인터페이스나 슈퍼클래스는 다시 로드되지 않습니다.
위 코드가 제대로 작동하려면 물론 인터페이스나 슈퍼 클래스가 상위 클래스에 의해 로드되도록 자체 클래스 로더를 구현해야 합니다. 클래스 로더가 MyObject를 로드하라는 요청을 받으면 MyObjectInterface 인터페이스 또는 MyObjectSuperclass 클래스도 로드하라는 요청을 받게 됩니다. 이는 MyObject 클래스에서 내부적으로 참조하기 때문입니다. 클래스 로더는 인터페이스 또는 수퍼클래스를 로드한 동일한 클래스 로더에 클래스 로딩을 위임해야 합니다.

类加载器加载/重新加载示例

上文包含了很多内容。让我们看一下简单的示例。下面是一个简单的ClassLoader子类。注意它如何将类加载委托给它的父类,除了它想要重装的一个类之外。如果类加载被委派给了它的父类,它以后将不能被重新加载。记住,一个类只能被同一个ClassLoader实例加载。
如前所述,这只是一个示例,它显示了类加载器的行为的基本知识。这并不是一个你的类加载器的生产就绪的模板。你的类加载器可能并不仅限于一个类,可能是一个你想要重新加载的类的集合。此外,你也不能硬编码class path。

public class MyClassLoader extends ClassLoader{

    public MyClassLoader(ClassLoader parent) {
        super(parent);
    }

    public Class loadClass(String name) throws ClassNotFoundException {
        if(!"reflection.MyObject".equals(name))
                return super.loadClass(name);

        try {
            String url = "file:C:/data/projects/tutorials/web/WEB-INF/" +
                            "classes/reflection/MyObject.class";
            URL myUrl = new URL(url);
            URLConnection connection = myUrl.openConnection();
            InputStream input = connection.getInputStream();
            ByteArrayOutputStream buffer = new ByteArrayOutputStream();
            int data = input.read();

            while(data != -1){
                buffer.write(data);
                data = input.read();
            }

            input.close();

            byte[] classData = buffer.toByteArray();

            return defineClass("reflection.MyObject",
                    classData, 0, classData.length);

        } catch (MalformedURLException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }

        return null;
    }

}

下面是使用MyClassLoader的示例:

public static void main(String[] args) throws
    ClassNotFoundException,
    IllegalAccessException,
    InstantiationException {

    ClassLoader parentClassLoader = MyClassLoader.class.getClassLoader();
    MyClassLoader classLoader = new MyClassLoader(parentClassLoader);
    Class myObjectClass = classLoader.loadClass("reflection.MyObject");

    AnInterface2       object1 =
            (AnInterface2) myObjectClass.newInstance();

    MyObjectSuperClass object2 =
            (MyObjectSuperClass) myObjectClass.newInstance();

    //create new class loader so classes can be reloaded.
    classLoader = new MyClassLoader(parentClassLoader);
    myObjectClass = classLoader.loadClass("reflection.MyObject");

    object1 = (AnInterface2)       myObjectClass.newInstance();
    object2 = (MyObjectSuperClass) myObjectClass.newInstance();

}

reflection.MyObject类是由自定义类加载器加载的。注意,它是如何继承一个超类、实现一个接口的。这只是为了这个例子。在你的代码中,只需要两个中的一个,继承超类或实现接口。

public class MyObject extends MyObjectSuperClass implements AnInterface2{
    //... body of class ... override superclass methods
    //    or implement interface methods
}

위 내용은 Java 동적 클래스 로딩 및 재로딩에 대한 자세한 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명
이 기사는 segmentfault思否에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제
JVM이 Java 코드와 기본 운영 체제 사이의 중개자 역할을하는 방법을 설명하십시오.JVM이 Java 코드와 기본 운영 체제 사이의 중개자 역할을하는 방법을 설명하십시오.Apr 29, 2025 am 12:23 AM

JVM은 Java 코드를 기계 코드로 변환하고 리소스를 관리하여 작동합니다. 1) 클래스로드 : .class 파일을 메모리에로드하십시오. 2) 런타임 데이터 영역 : 메모리 영역 관리. 3) 실행 엔진 : 해석 또는 컴파일 바이트 코드. 4) 로컬 메소드 인터페이스 : JNI를 통해 운영 체제와 상호 작용합니다.

Java의 플랫폼 독립성에서 JVM (Java Virtual Machine)의 역할을 설명하십시오.Java의 플랫폼 독립성에서 JVM (Java Virtual Machine)의 역할을 설명하십시오.Apr 29, 2025 am 12:21 AM

JVM을 통해 Java는 플랫폼을 가로 질러 실행할 수 있습니다. 1) JVM 하중, 검증 및 바이트 코드를 실행합니다. 2) JVM의 작업에는 클래스 로딩, 바이트 코드 검증, 해석 실행 및 메모리 관리가 포함됩니다. 3) JVM은 동적 클래스 로딩 및 반사와 같은 고급 기능을 지원합니다.

Java 응용 프로그램이 다른 운영 체제에서 올바르게 실행되도록하는 데 어떤 조치를 취 하시겠습니까?Java 응용 프로그램이 다른 운영 체제에서 올바르게 실행되도록하는 데 어떤 조치를 취 하시겠습니까?Apr 29, 2025 am 12:11 AM

Java 응용 프로그램은 다음 단계를 통해 다른 운영 체제에서 실행할 수 있습니다. 1) 파일 또는 경로 클래스를 사용하여 파일 경로를 처리합니다. 2) system.getenv ()를 통해 환경 변수를 설정하고 얻습니다. 3) Maven 또는 Gradle을 사용하여 종속성 및 테스트를 관리하십시오. Java의 크로스 플랫폼 기능은 JVM의 추상화 계층에 의존하지만 여전히 특정 운영 체제 별 기능의 수동 처리가 필요합니다.

Java가 플랫폼 별 구성 또는 튜닝이 필요한 영역이 있습니까?Java가 플랫폼 별 구성 또는 튜닝이 필요한 영역이 있습니까?Apr 29, 2025 am 12:11 AM

Java는 다른 플랫폼에서 특정 구성 및 튜닝이 필요합니다. 1) 힙 크기를 설정하기 위해 -xms 및 -xmx와 같은 JVM 매개 변수를 조정하십시오. 2) 병렬 GC 또는 G1GC와 같은 적절한 쓰레기 수집 전략을 선택하십시오. 3) 다른 플랫폼에 적응하도록 기본 라이브러리를 구성하십시오. 이러한 측정을 통해 Java 응용 프로그램은 다양한 환경에서 가장 잘 수행 할 수 있습니다.

Java 개발에서 플랫폼 별 문제를 해결하는 데 도움이되는 몇 가지 도구 또는 라이브러리는 무엇입니까?Java 개발에서 플랫폼 별 문제를 해결하는 데 도움이되는 몇 가지 도구 또는 라이브러리는 무엇입니까?Apr 29, 2025 am 12:01 AM

OSGI, APACHECMONSLANG, JNA, andJVMOPTIONSAREEFFICEPIPERINGLINGPLATFORM-SPECIFICCHALLENGENJAVA.1) OSGIMANAGESDENCENTENCENDISONDISOLATESCOMPONENTS.2) APACHECOMMONSLANGPROVIDEUTIOMITIONFUCTIONS.3) JNAALLOWSCALLINGNATIVECODE.4) JNAALLOWSCALTINGBEHAV

JVM은 다른 플랫폼에서 쓰레기 수집을 어떻게 관리합니까?JVM은 다른 플랫폼에서 쓰레기 수집을 어떻게 관리합니까?Apr 28, 2025 am 12:23 AM

jvmmanagesgarbageCollectionAcrossplatformSefficialthegendercationalStrationallySticallySticallySuciationalStrationalSproachandAptingToosandHardwaredifferences.ITEMPLOYSVARIOUSCOLLECTORSLIKESERIAL, PARALING, CMS, 및 G1, 각각의 소지 firedFferentscenarios.performanceCanbetwithflags-xex : xa

Java 코드가 수정없이 다른 운영 체제에서 실행할 수있는 이유는 무엇입니까?Java 코드가 수정없이 다른 운영 체제에서 실행할 수있는 이유는 무엇입니까?Apr 28, 2025 am 12:14 AM

Java의 "Write Onge, Run Everywhere"철학은 JVM (Java Virtual Machine)에서 구현되므로 Java Code는 수정없이 다른 운영 체제에서 실행할 수 있습니다. 컴파일 된 Java Bytecode와 운영 체제 사이의 중개자로서 JVM은 바이트 코드를 특정 시스템 지침으로 변환하여 프로그램이 JVM이 설치된 모든 플랫폼에서 독립적으로 실행될 수 있도록합니다.

플랫폼 독립성을 강조하는 Java 프로그램을 컴파일하고 실행하는 프로세스를 설명하십시오.플랫폼 독립성을 강조하는 Java 프로그램을 컴파일하고 실행하는 프로세스를 설명하십시오.Apr 28, 2025 am 12:08 AM

Java 프로그램의 편집 및 실행은 Bytecode 및 JVM을 통해 플랫폼 독립성을 달성합니다. 1) Java 소스 코드를 작성하여 바이트 코드로 컴파일하십시오. 2) JVM을 사용하여 모든 플랫폼에서 바이트 코드를 실행하여 코드가 플랫폼에서 실행되도록합니다.

See all articles

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

뜨거운 도구

MinGW - Windows용 미니멀리스트 GNU

MinGW - Windows용 미니멀리스트 GNU

이 프로젝트는 osdn.net/projects/mingw로 마이그레이션되는 중입니다. 계속해서 그곳에서 우리를 팔로우할 수 있습니다. MinGW: GCC(GNU Compiler Collection)의 기본 Windows 포트로, 기본 Windows 애플리케이션을 구축하기 위한 무료 배포 가능 가져오기 라이브러리 및 헤더 파일로 C99 기능을 지원하는 MSVC 런타임에 대한 확장이 포함되어 있습니다. 모든 MinGW 소프트웨어는 64비트 Windows 플랫폼에서 실행될 수 있습니다.

SublimeText3 영어 버전

SublimeText3 영어 버전

권장 사항: Win 버전, 코드 프롬프트 지원!

SublimeText3 Linux 새 버전

SublimeText3 Linux 새 버전

SublimeText3 Linux 최신 버전

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

Atom Editor Mac 버전 다운로드

Atom Editor Mac 버전 다운로드

가장 인기 있는 오픈 소스 편집기