Tomcat WebappClassLoader 类加载机制源码
2017-09-06 本文已影响240人
jijs
需要了解的知识JVM 类加载机制深入浅出
tomcat中的ClassLoader
![](https://img.haomeiwen.com/i2843224/7ec3346d4790ba08.png)
- 启动类加载器(BootStrap ClassLoader):引导类装入器是用本地代码实现的类装入器,它负责将 jdk中jre/lib下面的核心类库或-Xbootclasspath选项指定的jar包加载到内存中。由于引导类加载器涉及到虚拟机本地实现细节,开发者无法直接获取到启动类加载器的引用,所以不允许直接通过引用进行操作。
- 扩展类加载器(Extension ClassLoader):扩展类加载器是由Sun的ExtClassLoader(sun.misc.Launcher$ExtClassLoader)实现的。它负责将jdk中jre/lib/ext或者由系统变量-Djava.ext.dir指定位置中的类库加载到内存中。开发者可以直接使用标准扩展类加载器。
- 系统类加载器(System ClassLoader):系统类加载器是由 Sun的 AppClassLoader(sun.misc.Launcher$AppClassLoader)实现的。它负责将系统类路径java -classpath或-Djava.class.path变量所指的目录下的类库加载到内存中。开发者可以直接使用系统类加载器。
- StandardClassLoader 负责加载tomcat容器相关的类
- WebappClassLoader 是每个web项目对应一个WebappClassLoader。这样做的目的是每个项目中都会有相同的类(package+classname),而类的内容不一样。这样每个项目一个WebappClassLoader可以达到隔绝项目类冲突的问题。
下面我们主要讲解下WebappClassLoader的类加载机制
第一步
![](https://img.haomeiwen.com/i2843224/f679f09ba41ca14a.png)
首先调用findLoaderClass0() 方法检查WebappClassLoader中是否加载过此类。
![](https://img.haomeiwen.com/i2843224/72bb6adbb9efe4d9.png)
WebappClassLoader 加载过的类都存放在 resourceEntries 缓存中。
protected final Map<String, ResourceEntry> resourceEntries = new ConcurrentHashMap<>();
第二步
![](https://img.haomeiwen.com/i2843224/3561d7fa896109a3.png)
如果第一步没有找到,则继续检查JVM虚拟机中是否加载过该类。
调用ClassLoader的findLoadedClass() 方法检查
![](https://img.haomeiwen.com/i2843224/b795515f87485878.png)
第三步
![](https://img.haomeiwen.com/i2843224/48e6ca1a5df2b4ec.png)
如果前两步都没有找到,则使用AppClassloader加载该类(也就是当前JVM的ClassPath)
第四步
![](https://img.haomeiwen.com/i2843224/77b5c70698eff73a.png)
如果前三步都没找到,通过filter() 方法检查该类是否在定义的包名下,如果在则通过 StandardClassLoader类加载。
![](https://img.haomeiwen.com/i2843224/465710d8bab0473e.png)
![](https://img.haomeiwen.com/i2843224/0aac252ccc260425.png)
![](https://img.haomeiwen.com/i2843224/b5d6fbcb571d92a2.png)
第五步
![](https://img.haomeiwen.com/i2843224/836f403a3e902bc5.png)
如果前4步都没有找到,将由WebappClassLoader来加载。
![](https://img.haomeiwen.com/i2843224/63751544d20510a3.png)
从当前的工程的Web-INF/classes 目录下查找
![](https://img.haomeiwen.com/i2843224/1e214f9f0ad85d31.png)
如果找到,则创建ResourceEntry对象,保存这个类的元信息,并把他保存在WebappClassLoader的resourceEntries中,便于下次查找。
想了解更多精彩内容请关注我的公众号
![](https://img.haomeiwen.com/i2843224/af50fe51e979ebd0.png)