56

当我提供数据库 URL 时,为什么它说空URL 并在异常中给出一个空的 ' ' 类?

我正在尝试在使用 Tomcat 时通过 servlet 连接到 derby 数据库。当 servlet 运行时,我得到以下异常:

org.apache.tomcat.dbcp.dbcp.SQLNestedException: Cannot create JDBC driver of class '' for connect URL 'null'

at org.apache.tomcat.dbcp.dbcp.BasicDataSource.createConnectionFactory(BasicDataSource.java:1452)
at org.apache.tomcat.dbcp.dbcp.BasicDataSource.createDataSource(BasicDataSource.java:1371)
at org.apache.tomcat.dbcp.dbcp.BasicDataSource.getConnection(BasicDataSource.java:1044)

at servlets.servlet_1.doGet(servlet_1.java:23) // ---> Marked the statement in servlet

at javax.servlet.http.HttpServlet.service(HttpServlet.java:621)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:722)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:304)
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:210)
at org.netbeans.modules.web.monitor.server.MonitorFilter.doFilter(MonitorFilter.java:393)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:243)
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:210)
at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:224)
at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:169)
at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:472)
at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:168)
at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:100)
at org.apache.catalina.valves.AccessLogValve.invoke(AccessLogValve.java:929)
at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:118)
at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:405)
at org.apache.coyote.http11.AbstractHttp11Processor.process(AbstractHttp11Processor.java:964)
at org.apache.coyote.AbstractProtocol$AbstractConnectionHandler.process(AbstractProtocol.java:515)
at org.apache.tomcat.util.net.JIoEndpoint$SocketProcessor.run(JIoEndpoint.java:302)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:885)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:907)
at java.lang.Thread.run(Thread.java:619)
Caused by: java.lang.NullPointerException
at sun.jdbc.odbc.JdbcOdbcDriver.getProtocol(JdbcOdbcDriver.java:507)
at sun.jdbc.odbc.JdbcOdbcDriver.knownURL(JdbcOdbcDriver.java:476)
at sun.jdbc.odbc.JdbcOdbcDriver.acceptsURL(JdbcOdbcDriver.java:307)
at java.sql.DriverManager.getDriver(DriverManager.java:253)
at org.apache.tomcat.dbcp.dbcp.BasicDataSource.createConnectionFactory(BasicDataSource.java:1437)
... 24 more

小服务程序:

package servlets;

import java.io.IOException;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import javax.naming.Context;
import javax.naming.InitialContext;
import javax.servlet.http.*;
import javax.servlet.*;
import javax.sql.DataSource;

public class servlet_1 extends HttpServlet {

    @Override 
    public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        try {
           // String queryString = request.getQueryString();
            System.out.println("!!!!!!!!!!!!!!!!!!!");
            Context initContext = new InitialContext();
            Context envContext = (Context)initContext.lookup("java:comp/env");
            DataSource ds = (DataSource)envContext.lookup("jdbc/PollDatasource");
            Connection connection = ds.getConnection(); // -->LINE 23
            String sqlQuery = "select * from PollResult";
            PreparedStatement statement = connection.prepareStatement(sqlQuery);
            ResultSet set = statement.executeQuery();
            System.out.println("after the final statement");
        } catch (Exception exc) {
            exc.printStackTrace();
        }
    }
}

这是什么例外?为什么我会得到这个异常?

我在context.xmlTomcat 中添加了以下标签:

<Resource name="jdbc/PollDatasource" auth="Container" type="javax.sql.DataSource"
driverClassName="org.apache.derby.jdbc.EmbeddedDriver"
url="jdbc:derby://localhost:1527/poll_database;create=true"
username="suhail" password="suhail"
maxActive="20" maxIdle="10" maxWait="-1" />

这在web.xml

<resource-ref>
  <description>my connection</description>
  <res-ref-name>jdbc/PollDatasource</res-ref-name>
  <res-type>javax.sql.DataSource</res-type>
  <res-auth>Container</res-auth>
</resource-ref>

我在哪里犯错?

显示数据库 URL 的图像..

在此处输入图像描述

注意:在@Bryan Pendleton 的回答之后,我将驱动程序更改为org.apache.derby.jdbc.ClientDriver但我得到了同样的异常。

4

11 回答 11

37

我看不出有什么明显的错误,但也许另一种方法可以帮助您调试它?

您可以尝试在每个应用程序上下文中指定您的数据源,而不是在全局 tomcat 中指定。

您可以通过创建 src/main/webapp/META-INF/context.xml 来做到这一点(我假设您使用的是标准的 maven 目录结构 - 如果不是,那么 META-INF 文件夹应该是您的同级WEB-INF 目录)。META-INF/context.xml 文件的内容类似于:

<?xml version="1.0" encoding="UTF-8"?>

<Context [optional other attributes as required]>

<Resource name="jdbc/PollDatasource" auth="Container"
          type="javax.sql.DataSource" driverClassName="org.apache.derby.jdbc.ClientDriver"
          url="jdbc:derby://localhost:1527/poll_database;create=true"
          username="suhail" password="suhail" maxActive="20" maxIdle="10" maxWait="-1"/>
</Context>

显然,路径和 docBase 需要与您的应用程序的具体细节相匹配。

使用这种方法,您不必在 Tomcat 的 context.xml 文件中指定数据源详细信息。虽然,如果您有多个应用程序与同一个数据库通信,那么您的方法更有意义。

无论如何,试一试,看看它是否有什么不同。它可能会为我们提供有关您的方法出了什么问题的线索。

于 2012-07-21T00:42:21.090 回答
11

几个修复:

  1. 为您的环境使用正确的驱动程序类名称:如果您使用的是进程外 Derby 服务器,那么您需要 ClientDriver(并且需要使用 derbyclient.jar)、主机名和端口等。如果您想要一个 in-处理 Derby 服务器,然后您需要 derby.jar、EmbeddedDriver 和适用于嵌入式数据库的 URL。

  2. 仅将驱动程序 JAR 文件放在 Tomcat 的lib/目录中。

  3. 不要在 Tomcat 中放任何东西conf/context.xml:真的没有理由这样做。相反,使用你的 webappMETA-INF/context.xml来定义你的<Resource>.

错误的"Cannot create JDBC driver of class '' for connect URL 'null'发生通常是因为 JDBC 驱动程序不在正确的位置(或者在太多的地方,比如 Tomcat 的lib/目录,但也在 webapp 的WEB-INF/lib/目录中)。请确认您在正确的位置拥有正确的驱动程序 JAR 文件。

于 2012-07-22T21:23:43.673 回答
7

这两件事不匹配:

driverClassName="org.apache.derby.jdbc.EmbeddedDriver"
url="jdbc:derby://localhost:1527/poll database;create=true"

如果您使用 EmbeddedDriver,您的 URL 不应包含网络语法。

相反,如果您使用网络语法,则需要使用 ClientDriver。

http://db.apache.org/derby/docs/10.8/getstart/rgsquck35368.html

于 2012-07-17T17:53:39.590 回答
7

我遇到了这个问题,因为我context.xml走错了路:

./src/main/resources/META-INF/context.xml

正确的路径是:

./src/main/webapp/META-INF/context.xml
于 2015-12-01T11:50:00.353 回答
3

如果您使用的是 eclipse,您应该从 eclipse 包资源管理器中创建的服务器项目修改 context.xml。在eclipse中使用tomcat时,它是唯一有效的,其他的被忽略或覆盖

于 2013-11-25T09:50:47.717 回答
2

Context envContext = (Context)initContext.lookup("java:comp/env");

不是:Context envContext = (Context)initContext.lookup("java:/comp/env");

于 2012-07-17T06:02:21.243 回答
1

您是否尝试仅在context.xml

<Resource name="jdbc/PollDatasource" auth="Container" type="javax.sql.DataSource"
driverClassName="org.apache.derby.jdbc.EmbeddedDriver"
url="jdbc:derby://localhost:1527/poll_database;create=true"
username="suhail" password="suhail"
maxActive="20" maxIdle="10" maxWait="-1" />

并从中删除<resource-ref>部分web.xml

在一个项目中,我看到没有<resource-ref>部分的配置web.xml并且它有效。

这是一个有根据的猜测,但我认为命名资源的<resource-ref>声明可能会覆盖具有相同名称的资源声明,并且声明中缺少两者,因此缺少该属性的 NPE。JNDIjdbc/PollDatasourceweb.xmlcontext.xmlweb.xmldriverClassNameurl

于 2012-07-19T14:09:36.763 回答
1

就我而言,我解决了编辑 [tomcat]/Catalina/localhost/[mywebapp_name].xml 而不是 META-INF/context.xml 的问题。

于 2013-07-01T15:11:35.397 回答
1

问题也可能是由于SQL driver您的 Tomcat 安装目录中缺少。

我不得不mysql-connector-java-5.1.23-bin.jarapache-tomcat-9.0.12/lib/文件夹中。

https://dev.mysql.com/downloads/connector/j/

于 2018-09-16T10:22:46.470 回答
0

如果您使用的是嵌入式驱动程序,则 connectString 只是

jdbc:derby:databaseName 

(选项如;create=true;user=xxx 等)。

如果您使用客户端驱动程序,连接字符串可以保持原样,但如果更改驱动程序没有结果...请原谅这个问题,但您是否 100% 确定您已按照Derby 教程启动了 Derby 网络服务器?

于 2012-07-20T14:48:52.690 回答
0

我在针对 Oracle 使用 Tomcat 时遇到了类似的问题。我确实在光盘上的 META-INF 目录中有 context.xml。但是这个文件没有显示在 eclipse 项目中。一个简单的 F5 刷新和 context.xml 文件出现,eclipse 发布了它。一切都过去了。希望这可以帮助某人。

尝试在eclipse中按F5

于 2014-06-09T16:05:05.590 回答