我们一直迷惹为什么我们放在web-inf/classes和web-in/lib下的类和包怎会被tomcat找到执行,这也是我们为什么常常把自己项目开发的
类和第三方包放在这两个地方,以前我们一直认为java只从classpath中加载类,或者dir.ext中加载类及lib中加载系统类包。
但是tomcat是否有自己的加载方式,比如,tomcat能加载自己目下的核心类包,tomcat的核心类包放在apache-tomcat-6.0.14\lib
为什么会这样,原因在于tomcat定义自己的类加载器,tomcat类加载器所在的包为:
org.apache.catalina.loader
而其中就有.WebappClassLoader
这个类就分管web-inf/class 和web-inf/lib下的类加载。此类继承了,
public class WebappClassLoader
extends URLClassLoader
implements Reloader, Lifecycle
并重写的URLClassLoader中的方法,如:findClass方法等,findClass被重写后作用就是在web-inf/class 和web-inf/lib
查找用户提供的类全路径的类,如:dlp.oa.chen.Test
如果WebappClassLoader得到这样一个路径名后会把其转换成dlp/oa/chen/Test.class
然后加上起始前缀,也就是web-inf/classes
物理路径为:D:/apache-tomcat-6.0.14/webapps/dlp_oa/WEB-INF/classes
最后得到真正路径大致就是D:/apache-tomcat-6.0.14/webapps/dlp_oa/WEB-INF/classes/dlp/oa/chen/Test.class
因此WebappClassLoader会以这样的方式找到类文件并加载,
接下来会有几个问题:
1.那么类加载器怎么知道是在class中还是lib中呢。
2.如果class中和lib中存在同样的类,而且包也相同,结果会怎样?
3.我们加载的类中肯定有new 之类的方法,那加载中的类中的方法引用类,他是被谁加载的,他是怎样取得路径的呢?
1.很显然加载会先在lib中找,如果找不到再到class中去找。这里存在一个优先级的问题。
2.第二个问题依仍于第一个问题,很显然执行最先找到那个类。前题是类中的方法相同,不然可能无法真正引用,或引用时不对。
3.第三个问题,在测试中发现我们发现在用自定义类加载器加载的类中用new方式加载类的话一样是用加载他自的加载器加载,
也就是说new 方法会用他当前的类的加载器来加载新类,他也会调用我们的自定义的加载器方法findClass(name),只不过这里
我们要注决的就是name必须为全路径,也就包名加类名,在类加载器中,我们只能如下:
这里不能直接D:\\apache-tomcat-6.0.14\\webapps\\web_test\\WEB-INF\\classes\\com\\chen\\pub\\
因为new 时类会有包名,很显然基础路径最多只能到包名的前一层目录。因为程序中是用.号来引用包的。
如:com.chen.pub.Test,事实上在传成真实路径时会把.号变成\
-------------------------------------------------------------------------------------------------------------------
public static final String drive="D:\\apache-tomcat-6.0.14\\webapps\\web_test\\WEB-INF\\classes\\";
public static final String fileType=".class";
public Class findClass(String name){
byte[] data=loadClassData(name);
return defineClass(name,data,0,data.length);
}
private byte[] loadClassData(String name) {
if(name.indexOf(".")>=0)
name = name.replace('.', '\\');
System.out.println("====="+name);
FileInputStream fis=null;
byte[] data = null;
try {
fis = new FileInputStream(new File(drive+name+fileType));
ByteArrayOutputStream baos=new ByteArrayOutputStream();
int ch=0;
while((ch=fis.read())!=-1){
baos.write(ch);
}
data=baos.toByteArray();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return data;
}
分享到:
相关推荐
Tomcat内部定义了多个类加载器,包括Bootstrap、Extension、Application(分别对应JDK的启动类加载器、扩展类加载器和应用程序类加载器),以及Tomcat自定义的Common、Catalina、Shared和WebappX类加载器。...
在Java的Web应用开发中,WebappClassLoader是Tomcat等Servlet容器用于加载应用中类的特定类加载器。它负责加载Web应用程序的类和资源,确保每个Web应用都有自己的类加载器实例,从而实现类的隔离。 该问题可能涉及...
2. "Tomcat的类载入器以及源码分析.doc":详细介绍了Tomcat的类加载器,如WebappClassLoader,它如何根据不同的web应用程序加载类,以及如何避免类冲突。 3. "理解Tomcat的WebappClassLoader(web应用类加载器)一....
2. **CommonClassLoader**: 这是Tomcat的全局类加载器,负责加载服务器级别的类库,这些类库对所有Web应用可见。在双亲委派模型中,WebAppClassLoader会将找不到的类委托给CommonClassLoader。 3. **Catalina...
在示例代码中,我们看到`OneServlet`类的加载器是`WebappClassLoader`,接着是`StandardClassLoader`,然后是`AppClassLoader`,最后是`ExtClassLoader`和`Bootstrap ClassLoader`。`getParent()`方法用于获取父类...
双亲委派模型是 Java 类加载器的核心设计原则之一,它的目的是为了保证类的唯一性以及提高安全性。具体工作流程如下: 1. 如果一个类加载器收到了类加载请求,它首先不会自己去尝试加载这个类,而是先委托给父类...
在Tomcat中,类加载器有以下层次结构: 1. Bootstrap ClassLoader:由JVM自身加载JDK核心类。 2. Common ClassLoader:加载`$CATALINA_HOME/lib`目录下的JAR或类。 3. Server ClassLoader:加载`$CATALINA_BASE/...
在Tomcat 5.0.18的源码中,`org.apache.catalina.loader.WebappClassLoader`是Web应用类加载器的核心实现。它继承自`URLClassLoader`,并重写了`findClass()`和`loadClass()`方法,以支持上述的Web应用隔离策略和...
7. **使用隔离的类加载器**:通过设置不同的类加载策略,例如Web应用的`WebappClassLoader`,可以避免不同应用之间的jar冲突。 8. **社区支持**:在Apache Tomcat的官方论坛或Stack Overflow等技术社区提问,获取...
Tomcat使用自定义的类加载器来加载Web应用中的类,以实现不同应用之间的类隔离。主要有CommonClassLoader、SharedClassLoader和WebappClassLoader,它们按层次加载类。 5. **连接器(Connector)与协议处理器...
2. **类加载机制**:Tomcat使用自定义的类加载器,如`CommonClassLoader`、`WebappClassLoader`等,以实现不同Web应用间的隔离。 3. **请求处理**:` CoyoteAdapter`是Servlet容器与Tomcat网络接口(Coyote)之间的...
在Tomcat中,每个Web应用都有自己的类加载器(WebappClassLoader),这个类加载器负责加载该Web应用下的所有类。当一个类被加载后,类加载器通常不会重新加载它。然而,为了实现热部署,Tomcat采用了不同的策略。 ...
`WebappClassLoader`则负责加载应用的类和资源文件。 四、线程模型 Tomcat采用基于线程池的并发模型,`Executor`接口及其实现类`ThreadPoolExecutor`用于管理线程。默认情况下,Tomcat使用`AprLifecycleListener`...
Tomcat使用自定义的类加载器,遵循“父类加载优先”的原则,避免了类冲突问题。 在深入源码之前,开发者需要熟悉Java的多线程、网络编程和类加载器等基础知识。Tomcat6的源码阅读可以从以下几个关键点入手: 1. **...