Tomcat WebappClassLoader 加载外部lib包

Tomcat为什么要违背双亲委托机制

什么是双亲委托机制

指当一个类加载器收到一个类加载请求时,该类加载器首先会把请求委派给父类加载器。 每个类加载器都是如此,只有在父类加载器在自己的搜索范围内找不到指定类时,子类加载器才会尝试自己去加载。

Tomcat的类加载机制

  • Bootstrap 这个类加载器包含 Java 虚拟机提供的基本运行时类,以及系统扩展目录 ($JAVA_HOME/jre/lib/ext) 中存在的 JAR 文件中的任何类。 注意:一些 JVM 可能将其实现为多个类加载器,或者它可能根本不可见(作为类加载器)。
  • System 这个类装入器通常是从CLASSPATH环境变量的内容初始化的。所有这些类对Tomcat内部类和web应用程序都是可见的。但是,标准的 Tomcat 启动脚本($CATALINA_HOME/bin/catalina.sh 或 %CATALINA_HOME%\bin\catalina.bat)完全忽略了 CLASSPATH 环境变量本身的内容,而是从以下存储库构建系统类加载器:
    • $CATALINA_HOME/bin/bootstrap.jar 包含用于初始化Tomcat服务器的main()方法,以及它所依赖的类装入器实现类。
    • $CATALINA_BASE/bin/tomcat-juli.jar or $CATALINA_HOME/bin/tomcat-juli.jar 日志实现类。其中包括java.util.logging API的增强类,称为Tomcat JULI,以及Tomcat内部使用的Apache Commons Logging库的重命名副本。
    • $CATALINA_HOME/bin/commons-daemon.jar 来自Apache Commons Daemon项目的类。这个JAR文件不在catalina.bat|.sh脚本构建的CLASSPATH中,而是从bootstrap.jar的清单文件中引用。
  • Common 加载common.loader(conf/catalina.properties)下的classes文件、资源文件和jar包,这些类对Tomcat内部类和所有Web应用程序都是可见的。
  • Server 加载server.loader(conf/catalina.properties)下的classes文件、资源文件和jar包, 只对Tomcat内部可见,而对web应用程序完全不可见。
  • Shared 加载shared.loader(conf/catalina.properties)下的classes文件、资源文件和jar包,对所有web应用程序可见,并可用于在所有web应用程序之间共享代码。但是,对此共享代码的任何更新都需要Tomcat重新启动
  • WebappX 单个Tomcat实例中的每个Web应用程序创建的类加载器。加载Web应用程序的/Web-INF/Class目录中的所有未打包类和资源,再加上Web应用程序/Web-INF/lib目录下JAR文件中的类和资源,对此web应用程序都是可见的,而对其他类则不可见。

Tomcat 为什么要违背双亲委托机制

  • 双亲委托机制不能满足tomcat的业务需求。Webapp类加载器需要独立加载自身的class以及依赖的jar。例如webapp1依赖的spring版本为4.x,另一个依赖的spring版本为5.x. 如果使用双亲委托,那么spring的版本只能存在一个,没法满足这个需求。

  • tomcat 自身依赖的jar可能和项目依赖的jar有重合的地方,比如servlet.jar, 此时优先webapp的jar加载

WebappClassLoader 加载class 的过程

加载指定名称的类,使用以下算法搜索,直到找到并返回该类。如果不能找到该类,则返回ClassNotFoundException。

  1. 调用 findLoadedClass(String) 检查 class 是否已经被加载。如果已经加载,则返回相同的Class对象。

  2. 如果context.xml#<Loader delegate="true" />属性被设置为真,如果有的话,则调用父类加载器的loadClass()方法。

  3. 调用 findClass()来在我们本地定义的资源库中找到这个类。

  4. 如果有的话,调用我们父类加载器的loadClass()方法。

如果使用上述步骤找到了这个类,并且resolve标志为真,那么这个方法将对产生的Class对象调用resolveClass(Class)

代码在org.apache.catalina.loader.WebappClassLoaderBase#loadClass

修改Tomcat源码使WebAppLoader能加载外部Lib

  1. 下载Tomcat源码,准备开发环境

    1. 安装ant工具

    2. 下载tomat源码

    cd ${tomcat.source}
    # 准备开发环境
    # 如果是eclipse 
    ant ide-eclipse
    # 如果是intellij
    ant ide-intellij 
    # 编译源码
    ant
    
  2. 修改catalina包

    org/apache/catalina/webresources/StandardRoot.java

    /***
    * loader xdo custom libs
    * @throws LifecycleException
    */
    protected void processWebInfXdoLib() {
    	WebResource webResource = getResource("/WEB-INF/classes/application.properties", false, false);
      if (webResource.exists() && webResource.isFile()) {
        try {
          Properties properties = new Properties();
          properties.load(webResource.getInputStream());
          String xdoLibs = properties.getProperty("xdo.libs");
          log.info("xdo lib value: " + xdoLibs);
          if (xdoLibs == null || "".equals(xdoLibs.trim())) {
            return;
          }
          String[] libsArray = xdoLibs.split(",");
          for (String lib : libsArray) {
            File file = new File(lib);
            if (file.isDirectory()) {
              List<File> fileList = Arrays.stream(file.listFiles())
                .filter(it -> it.getName().endsWith(".jar")).collect(Collectors.toList());
              log.info("xdo lib: " + lib + " size: " + fileList.size());
              for (File jarFile : fileList) {
                createWebResourceSet(ResourceSetType.CLASSES_JAR,
                                     "/WEB-INF/classes", jarFile.toURI().toURL(), "/");
              }
            }
          }
        }catch (IllegalArgumentException | IllegalStateException ex) {
          log.info("xdo not load application.properties. \n" + ex.getMessage());
        } catch (IOException ex) {
          log.error("xdo custom loader jar error!", ex);
        }
    }
    

    org/apache/catalina/webresources/StandardRoot.java

org/apache/catalina/loader/WebappClassLoaderBase.java

/**
     * Start the class loader.
     *
     * @throws LifecycleException if a lifecycle error occurs
     */
@Override
public void start() throws LifecycleException {

  state = LifecycleState.STARTING_PREP;

  WebResource[] classesResources = resources.getResources("/WEB-INF/classes");
  for (WebResource classes : classesResources) {
    if (classes.isDirectory() && classes.canRead()) {
      localRepositories.add(classes.getURL());
    }
  }
  WebResource[] jars = resources.listResources("/WEB-INF/lib");
  for (WebResource jar : jars) {
    if (jar.getName().endsWith(".jar") && jar.isFile() && jar.canRead()) {
      localRepositories.add(jar.getURL());
      jarModificationTimes.put(
        jar.getName(), Long.valueOf(jar.getLastModified()));
    }
  }

	WebResource resource = resources.getResource("/WEB-INF/classes/application.properties");
  if (resource.exists() && resource.isFile()) {
    try {
      Properties properties = new Properties();
      properties.load(resource.getInputStream());
      String xdoLibs = properties.getProperty("xdo.libs");
      if (xdoLibs != null && !"".equals(xdoLibs.trim())) {
        String[] libsArray = xdoLibs.split(",");
        for (String lib : libsArray) {
          File file = new File(lib);
          if (file.isDirectory()) {
            List<File> fileList = Arrays.stream(file.listFiles())
              .filter(it -> it.getName().endsWith(".jar")).collect(Collectors.toList());
            log.info("xdo lib: " + lib + " size: " + fileList.size());
            for (File jarFile : fileList) {
              localRepositories.add(jarFile.toURI().toURL());
            }
          }
        }
      } else {
        log.info("not setting xdo loader");
      }
    } catch (IllegalArgumentException | IllegalStateException ex) {
      log.info("xdo not load application.properties. \n" + ex.getMessage());
    } catch (IOException ex) {
      log.error("xdo loader jar error!", ex);
    }
  }

  state = LifecycleState.STARTED;
}

修改项目,打包时分离LIb

bootWar

task clearLib(type: Delete) {
    delete "$buildDir/libs/lib"
}

task copyLib(type: Copy) {
    from configurations.compileClasspath
  	// 这边的关系跟bootWar中的是反的,如果war中排出,这边就需要包含,如果war是包含,这边就需要排出
//    exclude("*xdo*.jar")
//    include("*.jar")
    into "$buildDir/libs/lib"

    from configurations.runtimeClasspath
    into "$buildDir/libs/lib"
}

bootWar {

//    rootSpec.include("*xdo*.jar")
  	// 指定排除的jar包
    rootSpec.exclude("*.jar")

    // lib目录的清除和复制任务
    dependsOn clearLib
    dependsOn copyLib
}

bootJar

// 清除lib
task clearLib(type: Delete) {
    delete "$buildDir/libs/lib"
}

// 拷贝lib
task copyLib(type: Copy) {
    from configurations.compileClasspath
    into "$buildDir/libs/lib"
}

bootJar {
    excludes = ["*.jar"]

    // lib目录的清除和复制任务
    dependsOn clearLib
    dependsOn copyLib

    // 指定依赖包的路径,运行时不再需要指定 java.ext.dir 或 loader.path 参数。
    manifest {
        attributes "Manifest-Version": 1.0,
                'Class-Path': configurations.compileClasspath.files.collect { "lib/$it.name" }.join(' ')
    }
}

这种不需要修改tomcat,直接打包即可

参考

https://tomcat.apache.org/tomcat-9.0-doc/class-loader-howto.html

https://tomcat.apache.org/tomcat-9.0-doc/building.html

原文地址:https://www.cnblogs.com/warrior/p/15330422.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


&lt;servlet&gt; &lt;servlet-name&gt;tomcatpooljsp&lt;/servlet-name&gt; &lt;jsp-file&gt;/WEB-INF/tomcatpool.jsp&lt;/jsp-file&gt; &lt;/servlet&gt; &lt;servlet-mapping&gt; &lt;servlet-name&gt;tomcatpooljsp&lt;/servlet-nam...
遵循Java Servlet 规范第4节中的建议 ,Apache Tomcat实现了系统地重新加载Java类的方法,以允许在不重新启动整个服务器的情况下更新应用程序的组件。 此功能对于开发非常重要,因为事实证明,随着服务器启动和重启时间的延长,这会严重浪费开发人员的时间。实际上,Java EE堆栈应用服务器的服务器重新启动时间很慢,这是Tomcat广泛用于个人和企业级项目的推动力之一。但是,即使Tomcat也无法 像运行时重新加载应用程序一样快地启动。通过仅重新加载隔离的应用程序的更改的类,开发人员..
JMX(Java管理扩展)是一项非常强大的技术,可让您管理,监视和配置Tomcat MBean。如果您是Tomcat管理员,那么您应该熟悉如何在tomcat中启用JMX来监视堆内存,线程,CPU使用率,类以及配置各种MBean。在本文中,我将讨论如何使用JConsole启用并连接到Tomcat。我假设您已经安装了Tomcat(如果没有);您可以参考安装指南。转到安装了Tomcat的路径 转到bin文件夹 将文件创建为“ setenv.sh” 使用vi编辑器修改文件并添加以下内容
总览介绍 建立 取得Java 获取TomCat 将TomCat安装为Windows服务 将TomCat设置为Linux服务(系统化) 使用Nginx作为反向代理 基本用法 手动启动和停止TomCat 验证TomCat服务器正在运行 服务静态文件 服务Java服务器页面(JSP) 修改设定 部署网络应用 使用管理网页界面 创建一个TomCat管理员用户 访问管理网络应用 管理网络应用 结论 参考链接介绍在最简单的概念中,To.
PSI Probe是Lambda Probe的社区驱动分支,使用相同的开源许可证(GPLv2)分发。它旨在替换和扩展Tomcat Manager,从而使管理和监视Apache Tomcat实例更加容易。与许多其他服务器监视工具不同,PSI Probe不需要对现有应用程序进行任何更改。它通过可访问Web的界面提供所有功能,只需将其部署到服务器即可使用。这些功能包括:请求:即使在每个应用程序的基础上,实时监视流量。 会话:浏览/搜索属性,查看上一个IP,到期,估计大小。 JSP:浏览,查看源代码,进
监视和管理Tomcat目录介绍 启用JMX远程 使用JMX远程Ant任务管理Tomcat JMXAccessorOpenTask-JMX打开连接任务 JMXAccessorGetTask:获取属性值Ant任务 JMXAccessorSetTask:设置属性值Ant任务 JMXAccessorInvokeTask:调用MBean操作Ant任务 JMXAccessorQueryTask:查询MBean Ant任务 JMXAccessorCreateTask:远程创建MBean Ant任
1.tomcat与jetty都是一种servlet引擎,他们都支持标准的servlet规范和javaEE规范
“The origin server did not find a current representation for the target resource...
Artifacts是maven中的一个概念,表示某个module要如何打包,例如war exploded、war、jar、ear等等这种打包形式;
使用 IDEA 编辑器开发项目十分便捷,这里介绍使用 IDEA 编辑器添加 Tomcat
这篇“servlet和tomcat的知识点有哪些”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅...
这篇文章主要讲解了“Tomcat管理平台实例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Tomcat管理平...
本文小编为大家详细介绍“tomcat虚拟主机怎么配置”,内容详细,步骤清晰,细节处理妥当,希望这篇“tomcat虚拟主机怎么配置”文章能帮助大家解决疑惑,下面跟
今天小编给大家分享一下tomcat相关配置与eclipse集成的方法的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家
这篇“Tomcat之web应用的目录组成结构是怎样的”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,
今天小编给大家分享一下tomcat目录映射的方法的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大...
这篇“tomcat的环境怎么配置”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文...
环境:tomcat:apache-tomcat-7.0.35 cactiEZ:10.1系统:centos5.6_x64一、配置tomcat服务器1、添加账号vim tomcat-users.xml 重启tomcat2、安装snmp协议yum...
一、 软环下载地址软件链接地址https://files.cnblogs.com/files/jinrf/openssl-1.0.2-latest.tar.gzhttps://files.cnblogs.com/files/jinrf/apr-util-1.6...