10

クラスローダーで遊んでいるときに、次の例外が発生しました。

Exception in thread "main" java.lang.ClassCastException: xxx.Singleton cannot be cast to xxx.Singleton

これは、クラスローダのインスタンスが別のクラスローダのクラスにキャストできないことを意味しますか?

"" セキュリティを使用しても、クラスローダーのおかげで 3 つのシングルトンをインスタンス化できるコードを確認してください。

public static void main(String[] args) throws Exception {
        URL basePath = new URL("file:/myMavenPath/target/classes/");

    Object instance = getClassInstance(Singleton.class);
    System.out.println(instance);
    //
    Object instance2 = getClassInstance(
            new URLClassLoader( new URL[]{basePath} , null )
                    .loadClass("my.Singleton")
    );
    System.out.println(instance2);
    //
    Object instance3 = getClassInstance(
            new URLClassLoader( new URL[]{basePath} , null )
                    .loadClass("my.Singleton")
    );
    System.out.println(instance3);

    // Only the 1st cast is ok
    Singleton testCast1 = (Singleton) instance;
    System.out.println("1st cast ok");
    Singleton testCast2 = (Singleton) instance2;
    System.out.println("2nd cast ok");
    Singleton testCast3 = (Singleton) instance3;
    System.out.println("3rd cast ok");
}

private static Object getClassInstance(Class clazz) throws Exception {
    Method method = clazz.getMethod("getInstance");
    method.setAccessible(true);
    return method.invoke(null);
}


class Singleton {

    private static final Singleton INSTANCE = new Singleton();

    public static Singleton getInstance() {
        return INSTANCE;
    }

    private Singleton() {
        Exception e = new Exception();
        StackTraceElement[] stackTrace = e.getStackTrace();
        if (!"<clinit>".equals(stackTrace[1].getMethodName())) {
            throw new IllegalStateException("You shall not instanciate the Singleton twice !",e);
        }
    }

    public void sayHello() {
        System.out.println("Hello World ! " + this);
    }

}
4

3 に答える 3

10

クラスローダー間でキャストすることはできません。クラスIDは、完全修飾名とクラスローダーで構成されます。ここでクラスアイデンティティのクライシス を確認してください。

于 2012-08-08T13:53:27.653 に答える
4

まさにその通りです。異なるクラスローダーによってロードされたクラス間でキャストを使用することはできません。

この質問、「クラスローダーを越えてキャストする」は、物事をより明確にするかもしれません...

于 2012-08-08T13:52:47.487 に答える
4

Yes, you are correct.

This often happens in OSGi projects, because of bad dependency management.

于 2012-08-08T13:48:27.613 に答える