繁体   English   中英

Java是否保证Object.getClass()== Object.getClass()?

[英]Does Java guarantee that Object.getClass() == Object.getClass()?

我真的是在这里指身份平等。

例如,以下内容将始终打印true吗?

System.out.println("foo".getClass() == "fum".getClass());

是的,类令牌是唯一的(对于任何给定的类加载器而言)。

即,您将始终在相同的类加载器领域中获得对相同物理对象的引用。 但是,不同的类加载器将加载不同的类令牌,同时,当两个不同的类加载器加载相同的类定义时,该类定义也被视为不同。

有关示例,请参见我的较早答案

对于X类的两个实例,

x1.getClass() == x2.getClass()

除非

x1.getClass().getClassLoader() == x2.getClass().getClassLoader()

注意: Class.getClassLoader()可能返回null,这意味着引导ClassLoader。

是。

返回的Class对象是被表示的类的静态同步方法锁定的对象。

如果可以返回多个实例,则

public static synchronized void doSomething() {..}

将不是线程安全的。

JVM规范中所述,每个类加载器都可以保证它:

首先,Java虚拟机确定是否已记录L是由N表示的类或接口的启动加载器。如果是,则此创建尝试无效,并且加载会引发LinkageError。

也就是说,如果类加载器(L)试图绕过默认的Class实例缓存,并且使JVM对于同一个类名(N)多次加载byte[]定义,则JVM将引发LinkageError

例如,实现一个每次调用loadClass(...)都会调用defineClass(...)的类加载器(绕过默认缓存):

public class ClassloaderTest {

    private static final byte[] CLASS_DEF = readClassBytes();

    private static byte[] readClassBytes() {
        try {
            InputStream is = ClassloaderTest.class.getResourceAsStream("ClassloaderTest.class");
            ByteArrayOutputStream buffer = new ByteArrayOutputStream();
            int nRead;
            byte[] data = new byte[16384];
            while ((nRead = is.read(data, 0, data.length)) != -1) {
                buffer.write(data, 0, nRead);
            }
            buffer.flush();
            return buffer.toByteArray();
        } catch (IOException ex) {
            throw new AssertionError();
        }
    }

    private static ClassLoader createNonCachingClassloader() {
        return new ClassLoader() {
            @Override
            public Class<?> loadClass(String name) throws ClassNotFoundException {
                if (name.equals("classloader.ClassloaderTest")) {
                    return defineClass(name, CLASS_DEF, 0, CLASS_DEF.length);
                } else {
                    return getParent().loadClass(name);
                }
            }
        };
    }

    public static void main(String[] args) throws Exception {
        ClassLoader cl = createNonCachingClassloader();
        Class<?> cl1 = cl.loadClass("classloader.ClassloaderTest");
        Class<?> cl2 = cl.loadClass("classloader.ClassloaderTest");
        System.out.println(cl1==cl2);
    }
}

这就是发生的情况:

Exception in thread "main" java.lang.LinkageError: loader (instance of  classloader/ClassloaderTest$1): attempted  duplicate class definition for name: "classloader/ClassloaderTest"
    at java.lang.ClassLoader.defineClass1(Native Method)
    at java.lang.ClassLoader.defineClass(ClassLoader.java:760)
    at java.lang.ClassLoader.defineClass(ClassLoader.java:642)
    at classloader.ClassloaderTest$1.loadClass(ClassloaderTest.java:53)
    at classloader.ClassloaderTest.main(ClassloaderTest.java:64)

干杯

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM