ClassName(自己写的一个类).class.getClassLoader()与Object.class.getClassLoader()好象是不同的,他们有什么区别?
又详细查了一下资料:
启动类装载器(Bootstrap Loader:C编写的),标准扩展类装载器(Extended Loader:java编写的),类路径装载器(AppClassLoader:java编写的)
由于启动类装载器(Bootstrap Loader)是用C编写的所以调用它的时候会返回null
测试代码:
[code="java"]
Object o=new Object();
Class c=o.getClass();
ClassLoader loader=c.getClassLoader();
System.out.println(loader);
[/code]
输出为null
新建一个类Test,测试它的加载器。
[code="java"]Test o=new Test();
Class c=o.getClass();
ClassLoader loader=c.getClassLoader();
System.out.println(loader);[/code]
输出为sun.misc.Launcher$AppClassLoader@19821f
结果与我预想的一样,那么如何验证我之前的言论呢,即[quote]而MyClass.class的加载器调用的是类路径装载器,当MyClass.class请求加载的时候,JVM会看启动类装载器,标准扩展类装载器这两个加载器有没有加载这个类,如果加载了那么就放弃当前的这个类加载器,由于之前没有使用启动类装载器,标准扩展类装载器加载这个类,所有JVM还会用类路径装载器加载[/quote]
我们做个实验,建一个类Test.java
[code="java"]
public class Test {
/**
* @param args
*/
public static void main(String[] args) {
/*Coffee coffee=new Coffee();
coffee.prepareRecipe();*/
Test o=new Test();
Class c=o.getClass();
ClassLoader loader=c.getClassLoader();
System.out.println(loader);
}
}
[/code]
编译之后copy到Java_home 的classes目录下,如果没有就建一个,之后把编译好的Test.class放到这个目录里。之后在cmd控制台java的bin目录下运行java -cp ./classes Test
看看结果,在我的机器上是null,之后在放到c盘根目录运行,发现是sun.misc.Launcher$AppClassLoader@18d107f
这就说明如果所加载的类在java的根目录下的某个目录运行,它调用的classLoader就是
BootstrapLoader 而其他目录就是类路径加载器了,而如果你想知道在哪个目录下会调用BootstrapLoader的话
用这句就可以看到
[code="java"]
System.out.println(System.getProperty("sun.boot.class.path"));
[/code]
当然用BootstrapLoader性能要好一点,毕竟是C写的呀
class.getClassLoader() returns which ClassLoader Object load it.
YourClass 可能是被什么 JarClassLoader啦加载的(纯属猜测), Object 可能是jdk较底层的什么ClassLoader加载的,某些时候可以自定义ClassLoader的,那么用YourClassLoader load 的SomeClass,所对应的 SomeClass.class.getClassLoader() 就是你YourClassLoader。
我一般都用 Thread.currentThread().getContextClassLoader() 来加载ClassPath Resource,一直没有深究,没啥问题,用SomeClass.class.getClassLoader() 加载往往找不到,也没管它。
classloader(类加载器)分为4中:
启动类装载器,标准扩展类装载器,类路径装载器和网络类装载器
这4中加载的优先级从高到底。
其中long包中的类被启动类装载器加载,而MyClass.class的加载器调用的是类路径装载器,当MyClass.class请求加载的时候,JVM会看启动类装载器,标准扩展类装载器这两个加载器有没有加载这个类,如果加载了那么就放弃当前的这个类加载器,由于之前没有使用启动类装载器,标准扩展类装载器加载这个类,所有JVM还会用类路径装载器加载,这时MyClass.class.getClassLoader()得到的就是类路径装载器,而Object使用的是启动类装载器,所以就是不同的。
loader是根据类的extends关系一级一级加载的。当然你要是用Object.class那就根loader