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

时间:2018-03-14 12:01:32

标签: java tomcat tomcat8

我正在尝试使用数据源从我的代码打开数据库连接。 webapp部署在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

环境详情

  • Java 1.8.0_161
  • Tomcat 8.5(Tomcat 9.0.6上的相同错误)
  • 操作系统:Windows 10 Pro 64位

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

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

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

1 个答案:

答案 0 :(得分:3)

在将相同内容发布到Tomcat的邮件列表后,我找到了答案。从讨论中看来,由于Tomcat中提供的内存泄漏修复,这种情况似乎正在发生。请参阅以下链接Tomcat Bug 60620

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

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

cairo_font_options_t *opts = cairo_get_font_options(some_cairo_context);
cairo_font_options_set_antialias(opts, CAIRO_ANTIALIAS_GRAY); // or _NONE
cairo_set_font_options(some_cairo_context, opts);
cairo_font_options_destroy(opts);