Tomcat 抛出错误 "java.lang.ClassNotFoundException: org.apache.naming.java.javaURLContextFactory"

Tomcat throwing error "java.lang.ClassNotFoundException: org.apache.naming.java.javaURLContextFactory"

我正在尝试使用数据源从我的代码中打开数据库连接。 Web 应用程序部署在 Tomcat。但是当打开连接时 Tomcat 抛出以下错误:

javax.naming.NoInitialContextException: Cannot instantiate class: org.apache.naming.java.javaURLContextFactory
at javax.naming.spi.NamingManager.getInitialContext(Unknown Source) ~[?:1.8.0_161]
at javax.naming.InitialContext.getDefaultInitCtx(Unknown Source) ~[?:1.8.0_161]
at javax.naming.InitialContext.init(Unknown Source) ~[?:1.8.0_161]
at javax.naming.InitialContext.<init>(Unknown Source) ~[?:1.8.0_161]
at com.profinch.fincluez.jdbc.JDBCConnection$ConnectionHandler.<init>(JDBCConnection.java:244) [classes/:?]
at com.profinch.fincluez.jdbc.JDBCConnection$ConnectionHandler.<init>(JDBCConnection.java:214) [classes/:?]
at com.profinch.fincluez.jdbc.JDBCConnection.getConnection(JDBCConnection.java:86) [classes/:?]
at com.profinch.fincluez.jdbc.RDBMSQueryEngine.loadData(RDBMSQueryEngine.java:190) [classes/:?]
at com.profinch.fincluez.jdbc.QueryEngineWrapper.loadData(QueryEngineWrapper.java:62) [classes/:?]
at com.profinch.fincluez.infra.loaders.ConfigLoaderQueryMode.loadData(ConfigLoaderQueryMode.java:64) [classes/:?]
at com.profinch.fincluez.infra.loaders.ConfigLoaderQueryMode.compute(ConfigLoaderQueryMode.java:82) [classes/:?]
at com.profinch.fincluez.infra.loaders.ConfigLoaderQueryMode.compute(ConfigLoaderQueryMode.java:1) [classes/:?]
at java.util.concurrent.RecursiveTask.exec(Unknown Source) [?:1.8.0_161]
at java.util.concurrent.ForkJoinTask.doExec(Unknown Source) [?:1.8.0_161]
at java.util.concurrent.ForkJoinPool$WorkQueue.runTask(Unknown Source) [?:1.8.0_161]
at java.util.concurrent.ForkJoinPool.runWorker(Unknown Source) [?:1.8.0_161]
at java.util.concurrent.ForkJoinWorkerThread.run(Unknown Source) [?:1.8.0_161]

Caused by: java.lang.ClassNotFoundException: org.apache.naming.java.javaURLContextFactory
at java.net.URLClassLoader.findClass(Unknown Source) ~[?:1.8.0_161]
at java.lang.ClassLoader.loadClass(Unknown Source) ~[?:1.8.0_161]
at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source) ~[?:1.8.0_161]
at java.lang.ClassLoader.loadClass(Unknown Source) ~[?:1.8.0_161]
at java.lang.Class.forName0(Native Method) ~[?:1.8.0_161]
at java.lang.Class.forName(Unknown Source) ~[?:1.8.0_161]
at com.sun.naming.internal.VersionHelper12.loadClass(Unknown Source) ~[?:1.8.0_161]
at com.sun.naming.internal.VersionHelper12.loadClass(Unknown Source) ~[?:1.8.0_161]
... 17 more

事情是我能够看到我的网络应用程序能够打开 3 个连接,然后从数据库加载属性和部分配置,然后正确关闭它们。我知道这一点,因为我有一个集中式 API,通过它可以打开和关闭所有连接。所以我一直在计算打开和关闭的连接数。但是从第 4 个连接开始我得到这个错误。我正在 运行 连接多个线程,每个线程都在打开和关闭连接,每个线程都因同样的错误而失败。

打开连接的代码:

InitialContext initialContext = new InitialContext();//throwing exception here
DataSource dataSource = (DataSource) initialContext.lookup("java:/comp/env/jdbc/OracleDS");
Connection l_conn = dataSource.getConnection ();

Tomcat 中的数据源详细信息:

<Resource name="jdbc/OracleDS" auth="Container" type="javax.sql.DataSource" driverClassName="oracle.jdbc.OracleDriver" username="test" password="test" url="jdbc:oracle:thin:@localhost:1521/sid" initialSize="5" maxTotal="15" defaultAutoCommit="false" factory="org.apache.tomcat.jdbc.pool.DataSourceFactory"/>

这个问题只出现在 Tomcat 中,当我在 Wildfly11/Glassfish 中部署我的代码并连接到它的数据源时,我没有遇到任何问题并且代码 运行 很顺利。

我浏览了一些帖子,有些人建议做下面的事情,但仍然遇到同样的错误:

Properties l_props = new Properties();
l_props.put(Context.INITIAL_CONTEXT_FACTORY, "org.apache.naming.java.javaURLContextFactory");
l_props.put(Context.URL_PKG_PREFIXES, "org.apache.naming");
InitialContext initialContext = new InitialContext(l_props);//Still throws Exception

环境详细信息

注意:我正在 运行 从 Tomcat 服务器而不是任何 IDE 并且不使用 JUnit 下载网络应用程序

[更新:2018 年 3 月 15 日]:我有一些设法缩小问题范围的方法。我正在 运行 使用 Fork Join Framework 并行执行多个任务,每个任务都会打开一个专用连接并将其关闭。任务未使用任何共享连接。所以当所有任务 运行 并行时会出现此错误。如果我通过将 ForkJoinPool 大小设置为 1 将代码依次更改为 运行 这些任务,则不会出现此错误。目前我有 5 个并行任务 运行ning,我的数据源连接池大小配置为 maxTotal=20。不确定发生了什么。

[更新 16-Match-2018]:如果有人有兴趣模拟这个问题,我已经在 Git 上上传了测试代码。 URL: Git Hub Repository

我在将相同的内容发布到 Tomcat 的邮件列表后找到了答案。从那里的讨论来看,这似乎是由于 Tomcat 中提供的内存泄漏修复而发生的。参考以下linkTomcat Bug 60620

以及从那里 link 编辑的各种线程。

我在创建初始上下文和 JNDI 查找之前编写了以下代码来解决此问题:

//Fix Start
Thread l_thread = Thread.currentThread();
l_thread.setContextClassLoader(this.getClass().getClassLoader());
//Fix End
initialContext  = new InitialContext();