java classloader回顾
以sun jdk为例
classloader主要负责的工作
find class:找到class
Links class:class和文件link
define class:将二进制文件转换为Class的一个实例
classloader的find class的方式
- findLoadedClass:首先看是否已经被当前的classloader加载过了
- parent.loadClass:调用parent加载器来load,如果没有parent loader,直接调用native的findBootStrapClass*方法来获取
- findClass:如果上述都没有找到,调用当前findClass
Class c = findLoadedClass(name);
if (c == null) {
try {
if (parent != null) {
c = parent.loadClass(name, false);
} else {
c = findBootstrapClass0(name);
}
} catch (ClassNotFoundException e) {
// If still not found, then invoke findClass in order
// to find the class.
c = findClass(name);
}
}
classloader的分类
- bootstraploader:native,jvm启动后启动,负责加载jre中的核心lib/class(包括下面两个classloader)。
- ExtClassLoader:由bootstraploader加载,负责加载jre中的ext包,parent=null
- AppClassLoader:由bootstraploader加载,负责加载classpath中的类,parent=ExtClassLoader
- UserClassLoader:自定义classloader,通过url指定资源位置
context classloader
实现parent classloader装载子classloader url内的class的需求jetty classloader
WebAppClassLoader.loadClass:
1. 查找已经在的class
Class<?> c= findLoadedClass(name);
2. 如果是parentfirst或者system_class并且不是server_class,则采用parentfirst
if (c == null && _parent!=null && (_context.isParentLoaderPriority() || system_class) && !server_class)
{
tried_parent= true;
try
{
c= _parent.loadClass(name);
if (Log.isDebugEnabled())
Log.debug("loaded " + c);
}
catch (ClassNotFoundException e)
{
ex= e;
}
}
3. 否则调用当前的findClass
if (c == null)
{
try
{
c= this.findClass(name);
}
catch (ClassNotFoundException e)
{
ex= e;
}
}
4. 当前木找到,找parent
if (c == null && _parent!=null && !tried_parent && !server_class )
c= _parent.loadClass(name);
5. resolve
if (resolve)
resolveClass(c);
如果是childfirst,则查找路劲是:
WebAppClassLoader-->bootstraploader-->ExtClassLoader-->AppClassLoader