Tomcat负责Web应用的类加载的是org.apache.catalina.loader.WebappClassLoader,它有几个比较重要的方法:findClass(),loadClass(),findClassInternal(),findResourceInternal().
WebappClassLoader类加载器被用来加载一个类的时候,loadClass()会被调
用,loadClass()则调用findClass()。后两个方法是WebappClassLoader的私有方法,findClass()调用
findClassInternal()来创建class对象,而findClassInternal()则需要
findResourceInternal()来查找.class文件。
通常自己实现类记载器只要实现findclass即可,这里为了实现特殊目的而override了loadClass().
下面是精简过的代码(去除了几乎全部关于log、异常和安全控制的代码):
public Class loadClass(String name, boolean resolve)
throws ClassNotFoundException {
Class clazz = null;
// (0) 先从自己的缓存中查找,有则返回,无则继续
clazz = findLoadedClass0(name);
if (clazz != null) {
if (resolve)
resolveClass(clazz);
return (clazz);
}
// (0.1) 再从parent的缓存中查找
clazz = findLoadedClass(name);
if (clazz != null) {
if (resolve)
resolveClass(clazz);
return (clazz);
}
// (0.2) 缓存中没有,则首先使用system类加载器来加载
clazz = system.loadClass(name);
if (clazz != null) {
if (resolve)
resolveClass(clazz);
return (clazz);
}
// 判断是否需要先让parent代理
boolean delegateLoad = delegate || filter(name);
// (1) 先让parent加载,通常delegateLoad == false,即这一步不会执行
if (delegateLoad) {
ClassLoader loader = parent;
if (loader == null)
loader = system;
clazz = loader.loadClass(name);
if (clazz == null) {
if (resolve)
resolveClass(clazz);
return (clazz);
}
}
// (2) delegateLoad == false 或者 parent加载失败,调用自身的加载机制
clazz = findClass(name);
if (clazz == null) {
if (resolve)
resolveClass(clazz);
return (clazz);
}
// (3) 自己加载失败,则请求parent代理加载
if (!delegateLoad) {
ClassLoader loader = parent;
if (loader == null)
loader = system;
clazz = loader.loadClass(name);
if (clazz == null) {
return (clazz);
}
}
throw new ClassNotFoundException(name);
}
public Class findClass(String name) throws ClassNotFoundException {
// 先试图自己加载类,找不到则请求parent来加载
// 注意这点和java默认的双亲委托模式不同
Class clazz = null;
clazz = findClassInternal(name);
if ((clazz == null) && hasExternalRepositories) {
synchronized (this) {
clazz = super.findClass(name);
}
}
if (clazz == null) {
throw new ClassNotFoundException(name);
}
return (clazz);
}
protected Class findClassInternal(String name) throws ClassNotFoundException {
if (!validate(name))
throw new ClassNotFoundException(name);
// 根据类名查找资源
String tempPath = name.replace('.', '/');
String classPath = tempPath + ".class";
ResourceEntry entry = null;
entry = findResourceInternal(name, classPath);
if (entry == null)
throw new ClassNotFoundException(name);
// 如果以前已经加载成功过这个类,直接返回
Class clazz = entry.loadedClass;
if (clazz != null)
return clazz;
// 以下根据找到的资源(.class文件)进行:1、定义package;2、对package安全检查;3、定义class,即创建class对象
synchronized (this) {
if (entry.binaryContent == null && entry.loadedClass == null)
throw new ClassNotFoundException(name);
// Looking up the package
String packageName = null;
int pos = name.lastIndexOf('.');
if (pos != -1)
packageName = name.substring(0, pos);
Package pkg = null;
if (packageName != null) {
pkg = getPackage(packageName);
// Define the package (if null)
if (pkg == null) {
// 定义package的操作,此处省略,具体参看源码
pkg = getPackage(packageName);
}
}
if (securityManager != null) {
// 安全检查操作,此处省略,具体参看源码
}
// 创建class对象并返回
if (entry.loadedClass == null) {
try {
clazz = defineClass(name, entry.binaryContent, 0, entry.binaryContent.length, new CodeSource(entry.codeBase, entry.certificates));
} catch (UnsupportedClassVersionError ucve) {
throw new UnsupportedClassVersionError(ucve.getLocalizedMessage() + " " + sm.getString("webappClassLoader.wrongVersion", name));
}
entry.loadedClass = clazz;
entry.binaryContent = null;
entry.source = null;
entry.codeBase = null;
entry.manifest = null;
entry.certificates = null;
} else {
clazz = entry.loadedClass;
}
}
return clazz;
}
protected ResourceEntry findResourceInternal(String name, String path) {
// 先根据类名从缓存中查找对应资源 ,有则直接返回
ResourceEntry entry = (ResourceEntry) resourceEntries.get(name);
if (entry != null)
return entry;
int contentLength = -1;
// 资源二进制数据长度
InputStream binaryStream = null;
// 资源二进制输入流
int jarFilesLength = jarFiles.length;
// classpath中的jar包个数
int repositoriesLength = repositories.length;
// 仓库数(classpath每一段称为repository仓库)
int i;
Resource resource = null;
// 加载的资源实体
boolean fileNeedConvert = false;
// 对每个仓库迭代,直接找到相应的entry,如果查找的资源是一个独立的文件,在这个代码块可以查找到相应资源,
// 如果是包含在jar包中的类,这段代码并不能找出其对应的资源
for (i = 0; (entry == null) && (i < repositoriesLength); i++) {
try {
String fullPath = repositories[i] + path;
// 仓库路径 加资源路径得到全路径
Object lookupResult = resources.lookup(fullPath);
// 从资源库中查找资源
if (lookupResult instanceof Resource) {
resource = (Resource) lookupResult;
}
// 到这里没有抛出异常,说明资源已经找到,现在构造entry对象
if (securityManager != null) {
PrivilegedAction dp = new PrivilegedFindResource(files[i], path);
entry = (ResourceEntry) AccessController.doPrivileged(dp);
} else {
entry = findResourceInternal(files[i], path);
// 这个方式只是构造entry并给其codebase和source赋值
}
// 获取资源长度和最后修改时间
ResourceAttributes attributes = (ResourceAttributes) resources.getAttributes(fullPath);
contentLength = (int) attributes.getContentLength();
entry.lastModified = attributes.getLastModified();
// 资源找到,将二进制输入流赋给binaryStream
if (resource != null) {
try {
binaryStream = resource.streamContent();
} catch (IOException e) {
return null;
}
// 将资源的最后修改时间加到列表中去,代码略去,参加源码
}
} catch (NamingException e) {
}
}
if ((entry == null) && (notFoundResources.containsKey(name)))
return null;
// 开始从jar包中查找
JarEntry jarEntry = null;
synchronized (jarFiles) {
if (!openJARs()) {
return null;
}
for (i = 0; (entry == null) && (i < jarFilesLength); i++) {
jarEntry = jarFiles[i].getJarEntry(path);
// 根据路径从jar包中查找资源
// 如果jar包中找到资源,则定义entry并将二进制流等数据赋给entry,同时将jar包解压到workdir.
if (jarEntry != null) {
entry = new ResourceEntry();
try {
entry.codeBase = getURL(jarRealFiles[i], false);
String jarFakeUrl = getURI(jarRealFiles[i]).toString();
jarFakeUrl = "jar:" + jarFakeUrl + "!/" + path;
entry.source = new URL(jarFakeUrl);
entry.lastModified = jarRealFiles[i].lastModified();
} catch (MalformedURLException e) {
return null;
}
contentLength = (int) jarEntry.getSize();
try {
entry.manifest = jarFiles[i].getManifest();
binaryStream = jarFiles[i].getInputStream(jarEntry);
} catch (IOException e) {
return null;
}
if (antiJARLocking && !(path.endsWith(".class"))) {
// 解压jar包代码,参见源码
}
}
if (entry == null) {
synchronized (notFoundResources) {
notFoundResources.put(name, name);
}
return null;
}
// 从二进制流将资源内容读出
if (binaryStream != null) {
byte[] binaryContent = new byte[contentLength];
// 读二进制流的代码,这里省去,参见源码
entry.binaryContent = binaryContent;
}
}
// 将资源加到缓存中
synchronized (resourceEntries) {
ResourceEntry entry2 = (ResourceEntry) resourceEntries.get(name);
if (entry2 == null) {
resourceEntries.put(name, entry);
} else {
entry = entry2;
}
}
return entry;
}
}
分享到:
相关推荐
3. "理解Tomcat的WebappClassLoader(web应用类加载器)一.doc":进一步探讨WebappClassLoader的工作机制,包括类查找策略、双亲委派模型的实现及其在多web应用环境下的应用。 4. "Tomcat源码分析(4)容器处理链接之...
Java中的类加载器通常分为三种:启动类加载器(Bootstrap ClassLoader)、扩展类加载器(Extension ClassLoader)和应用程序类加载器(Application ClassLoader)。它们遵循双亲委派模型,即较低层次的类加载器会将...
5. Web Application ClassLoader:每个Web应用都有自己独立的类加载器,加载`WEB-INF/classes`和`WEB-INF/lib`目录下的类和JAR。 **二、优化JAR包加载** 1. **减少重复JAR**:检查各个Web应用的`WEB-INF/lib`目录,...
在一些特定的应用场景下,如 Tomcat 服务器,会使用更为复杂的类加载器架构。例如,Tomcat 使用了多种类加载器来处理不同的类资源: 1. **CommonClassLoader**: - 负责加载 `/common/*` 目录中的类。 2. **...
这个要从热部署开始说起,因为tomcat提供了不必重启容器而只需重启web应用以达到热部署的功能,其实现是通过定义一个WebappClassLoader类加载器,当热部署时就将原来的类加载器废弃并重新实例化一个WebappCl
Tomcat的类加载机制遵循“父类加载优先”原则,通过`CommonLoader`、`CatalinaLoader`、`SharedLoader`和`WebappClassLoader`四个层次进行加载,保证了不同Web应用之间的类隔离。 8. **安全性** Tomcat支持多种...
在Tomcat中,每个Web应用都有自己的类加载器(WebappClassLoader),这个类加载器负责加载该Web应用下的所有类。当一个类被加载后,类加载器通常不会重新加载它。然而,为了实现热部署,Tomcat采用了不同的策略。 ...
它是Tomcat为每个web应用创建的特定类加载器,用于加载该应用的类。由于类加载器的委托机制,`WebappClassLoader`会先尝试加载类,如果找不到则由其父类加载器加载。这样可以确保web应用的类优先于容器的类。每个web...
- `-Dorg.apache.catalina.loader.WebappClassLoader.ENABLE_CLEAR_REFERENCES=false`:解决Tomcat重新加载Web应用时,不清除class文件中static和final变量的值而导致的潜在问题。 - `-Duser.timezone=GMT+08`:...