由于Tomcat可以一次加载多个Web应用程序,并且这些Web应用程序可以单独运行,并且不会互相干扰,因此它们可以在同一JVM中运行。因此,我对于Tomcat如何在同一JVM中处理Object范围感到非常困惑。
例如,我在两个不同的Webapp中都有一个单例对象,并且tomcat将为每个生成两个不同的单例对象。我一直认为单例对象在同一JVM中只有一个对象,但是在tomcat JVM中可能有两个或更多。
我已经阅读了有关ClassLoader的一些信息,Tomcat具有自己的WebAppClassLoader来加载webapp。这是否意味着这里的Object Scope是ClassLoader还是我错了。有人知道吗,或者可以给我一些有关tomcat工作存储器布局的信息吗?
所有的秘密都在这些ClassLoader实例背后。
ClassLoader
类的状态(如所有静态变量,字节码等)由加载该类的类加载器确定范围(在JVM中,该类通过其完全限定的名称标识,而类加载器则在加载该类。精确地确定范围,但将范围视为范围通常有助于更好地理解这一点。
因此,如果一个类由两个不同的类加载器加载,则该类在VM中存在两次,它具有两组静态字段,可以具有不同的字节码(如不同的方法实现)等等。请注意,即使两个对象的名称相同,也无法将它们强制转换。“普通” Java应用程序具有通过类加载器层次结构加载的所有类,并且每个类仅加载一次。
对于更复杂的方案,您将需要不同的行为。有时,您想将库与混乱的代码隔离开(例如eclipse中的插件或应用程序服务器中的Web应用程序)。
将您的程序与其他类隔离的基本思想是使用额外的类加载器加载那些类并进行大量反射。如果您想阅读有关内容,请查看有关ClassLoaders或OSGI的 Oracle文档。
Tomcat(以及许多其他Web容器/应用程序服务器)使用单独的ClassLoader层次结构加载应用程序。这样可以将所有类与其他(Web)应用程序隔离开,从而确保单例,不同的类版本以及所有这些东西不会冲突。