雄猫投掷错误"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

问题是我能够看到我的 web 应用程序能够打开 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

环境详细信息

  • Java 1.8.0_161
  • Tomcat
  • 8.5(在Tomcat 9.0.6上出现相同的错误)
  • 操作系统: 视窗 10 专业版 64 位

注意:我从Tomcat服务器而不是从任何IDE运行Web应用程序,并且不使用JUnit。

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

[更新16-Match-2018]:如果有人有兴趣模拟这个问题,我已经在Git上上传了测试代码。网址: Git 中心存储库

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

以及从那里链接的各种线程。

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

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

最新更新