首页 > 解决方案 > 根异常是 javax.naming.NameNotFoundException: Name jdbc not found in context "java:comp/env"

问题描述

我正在开发一个 Java Web 应用程序,并且正在部署在 Websphere Application Server 中,

我有一个类 Database.java,它返回与 DB2 数据库的 JDBC 连接,

        jndiName = "jdbc/TestDataSource";
        Context ctx = new InitialContext();
        envContext = (Context) ctx.lookup("java:comp/env");
        javax.sql.DataSource ds = (javax.sql.DataSource) envContext.lookup(jndiName);
        conn = ds.getConnection();

在执行操作的其他类中,创建 Database.java 类的对象并获取连接并执行 jdbc 操作,到目前为止它工作正常,但在最新开发中我必须在 servlet 类中引入一个线程,所以正在执行操作的其他类在后台线程中运行,并立即将控制权返回给 servlet。但是在实现这个应用程序之后,Websphere Application Server 崩溃了,并出现了下面提到的错误,但令人惊讶的是,它的 Tomcat 服务器工作得非常好。

PFB Web.xml

<resource-ref>
        <description>Connection-ConnectionPool</description>
        <res-ref-name>jdbc/TestDataSource</res-ref-name>
        <res-type>javax.sql.DataSource</res-type>
        <res-auth>Container</res-auth>
        <res-sharing-scope>Shareable</res-sharing-scope>
        <mapped-name>jdbc/TestDataSource</mapped-name>
</resource-ref>

错误:000000e9 SystemOut O NamingException - Database.getConnection(datasource) 中的错误:- javax.naming.ConfigurationException:无法完成对“java:”名称的 JNDI 操作,因为服务器运行时无法将操作线程与任何J2EE 应用程序组件。当使用“java:”名称的 JNDI 客户端未在服务器应用程序请求的线程上执行时,可能会发生这种情况。确保 J2EE 应用程序不会在静态代码块或由该 J2EE 应用程序创建的线程中对“java:”名称执行 JNDI 操作。此类代码不一定在服务器应用程序请求的线程上运行,因此不受“java:”名称上的 JNDI 操作的支持。[根异常是 javax.naming.NameNotFoundException:

标签: javamultithreadingwebspherejndiwebsphere-9

解决方案


该异常意味着您创建的线程没有 Java EE 应用程序的上下文,因此它无法知道要从哪个 Java EE 组件命名空间获取java:comp/env/jdbc/TestDataSource。请注意,您可以为不同的 Java EE 应用程序/组件定义相同的名称来表示不同的含义。有一个简单的方法可以解决这个问题,它遵循 Java EE 规范标准。不要创建自己的线程,而是使用 ManagedExecutorService(也是 Java EE 的一部分),它会自动将应用程序的上下文传播到托管线程,在这种情况下,应用服务器将能够正确解析查找。

这是一个例子:

ExecutorService executor = InitialContext.doLookup("java:comp/DefaultManagedExecutorService");

executor.submit(new Callable<Object>() {
    public Object call() throws Exception {
        DataSource ds = InitialContext.doLookup("java:comp/env/jdbc/TestDataSource");
        Connection conn = ds.getConnection();
        try {
           // ... do useful stuff with connection
        } finally {
           conn.close();
        }
        return result;
    }
});

推荐阅读