Java中一共有四个类加载器,之所以叫类加载器,是程序要用到某个类的时候,要用类加载器载入内存。
这四个类加载器分别为:Bootstrap ClassLoader、Extension ClassLoader、AppClassLoader和URLClassLoader,其中AppClassLoader在很多地方被叫做System ClassLoader。
Bootstrap ClassLoader是在JVM开始运行的时候加载java的核心类,是用C++编写的,它用来加载核心类库比如rt.jar,tools.jar等。
Extension ClassLoader是用来加载扩展类,即/lib/ext中的类。用的不多,比如dns,security什么的。
AppClassLoader用来加载Classpath的类,实际上我们还常用AppClassLoader来加载配置文件。spring可以用classpath做前缀,就是这个原理。
URLClassLoader用来加载网络上远程的类,不讨论。
按顺序是从大到小。 java 的class 加载采取委托模式:某个classloader 加载之前会先调用他上级来加载,只有上级找不到了,才会自己加载。这样做的第一个好处就是为了安全。比如,一定没有人能够伪造jre 标准类的java.lang.String。
载入class不等于new一个class。
在jdbc代码中有一句奇怪的。class.forName("xxxDriver");看驱动源码
public class Driver extends NonRegisteringDriver{
public Driver() throws SQLException{}
static {
try{
DriverManager.registerDriver(new Driver());
}
catch(SQLException E){
throw new RuntimeException("Can't register driver!");
}
}
}
Driver 中只有一个静态的方法把自己注册到driverManager中。那么就完全没必要new一个他的实例。载入就好了。但是想想还是不对,driver不是在web Server起来的时候就载入了吗?
最终结论:class的static块,调用ClassLoader.loadClass是不会执行的。只有在调用forName(String)载入class时执行。static永远只会执行一次。想来web服务器装载类的时候都是用的loadClass方法吧。
注:如果需要指定classLoader,使用自定义的classLoder加载,比如tomcat下面挂10个应用,可以在各自的线程中通过setContextClassLoader方法绑定各自的classLoader。
另:servlet的加载方式违背了委托模式。
详见tomcat的类加载机制:
http://www.blogjava.net/tomjamescn/archive/2009/06/23/283670.html