6

クラスローディングの概念に関して1つ質問があります。JVMで.classファイルを2回ロードする方法。また、これを実現するために作成したコードの抜粋も作成しています。

1)ローダー1コード

public class MyClassLoader extends ClassLoader {

    public MyClassLoader(){
        super(MyClassLoader.class.getClassLoader());
    }

    public Class loadClass(String classname){
        try {
            return super.loadClass(classname);
        } catch (ClassNotFoundException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return null;
    }
}

2)ローダー2コード

public class AnotherClassLoader extends ClassLoader {

    public AnotherClassLoader(){
        super(AnotherClassLoader.class.getClassLoader());
    }

    public Class loadClass(String classname){
        try {
            return super.loadClass(classname);
        } catch (ClassNotFoundException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return null;
    }
}

3)この2つの異なるクラスローダーを使用して、Aという名前のクラスをロードしています。オペレーションclassA1==newClassAはfalseを返すはずだと思います。コードは次のとおりです。

public static void main(String[] args) {
        MyClassLoader loader1 = new MyClassLoader();
        AnotherClassLoader newLoader = new AnotherClassLoader();
            System.out.println("Load with Custom Class Loader instance");
            Class classA1 = loader1.loadClass("com.hitesh.coreJava.A");
            System.out.println("Class Loader:::"+classA1.getClassLoader());
            Class newClassA = newLoader.loadClass("com.hitesh.coreJava.A");
            System.out.println("Class Loader:::"+newClassA.getClassLoader());
            System.out.println(classA1==newClassA);
            System.out.println(classA1.hashCode() + " , " + newClassA.hashCode());

    }

4)上記のコードを実行した結果:

カスタムクラスローダーインスタンスを使用してロードするクラスローダー:::sun.misc.Launcher$AppClassLoader@11b86e7クラスローダー:::sun.misc.Launcher$AppClassLoader@11b86e7true 1641745、1641745

これを説明していただけませんか

4

3 に答える 3

8

これを試して

public class Test1 {

    static class TestClassLoader1 extends ClassLoader {

        @Override
        public Class<?> loadClass(String name) throws ClassNotFoundException {
            if (!name.equals("Test1")) {
                return super.loadClass(name);
            }
            try {
                InputStream in = ClassLoader.getSystemResourceAsStream("Test1.class");
                byte[] a = new byte[10000];
                int len  = in.read(a);
                in.close();
                return defineClass(name, a, 0, len);
            } catch (IOException e) {
                throw new ClassNotFoundException();
            }
        }
    }


    public static void main(String[] args) throws Exception {
        Class<?> c1 = new TestClassLoader1().loadClass("Test1");
        Class<?> c2 = new TestClassLoader1().loadClass("Test1");
        System.out.println(c1);
        System.out.println(c2);
        System.out.println(c1 == c2);
    }
}

出力

class Test1
class Test1
false
于 2013-01-10T12:22:49.703 に答える
6

どちらのクラスローダーも、親のクラスローダーでルックアップを開始します (これがsuper()呼び出しの目的です)。したがって、実際にはどちらの場合もスーパークラスローダーがロードします。

これを試すことができます:

String pathToJar = "C:\\path\\to\\my.jar";
String className = "com.mypackage.ClassA";
URLClassLoader cl1 = new URLClassLoader(new URL[] { new URL(pathToJar) });
URLClassLoader cl2 = new URLClassLoader(new URL[] { new URL(pathToJar) });
Class<?> c1 = cl1.loadClass(className);
Class<?> c2 = cl2.loadClass(className);
System.out.println(c1);
System.out.println(c2);
System.out.println(c1==c2 ? "Parent classloader loads" : "Parent classloader does not load");
cl1.close();
cl2.close();

my.jarクラスパスにないことを確認してください。

于 2013-01-10T11:57:01.863 に答える
3

どちらの場合も、同じ ClassLoader を使用してロードを実行しています。2 つの ClassLoader がありますが、それぞれ AppClassLoader である同じ親 ClassLoader にデリゲートする super.loadClass() を呼び出すだけです。

于 2013-01-10T11:56:22.147 に答える