java - 获取 java.lang.NoClassDefFoundError : Could not initialize class oracle. jdbc.OracleDriver 异常

标签 java jdbc sybase

尝试使用 JDBC 驱动程序访问数据库时,我看到了一些奇怪的行为。这是代码片段:

LOGGER.debug("driver is " + driver);
try {
    Class.forName(driver);
    LOGGER.debug("got driver");
} catch (Throwable t) {
    LOGGER.debug("throwable getting driver " + driver);
    t.printStackTrace(System.out);
    throw t;
}

当我运行它时,这是我在堆栈跟踪中看到的内容。

08:20:00.417 [main] DEBUG - driver is com.sybase.jdbc4.jdbc.SybDriver
08:20:00.604 [main] DEBUG - throwable getting driver com.sybase.jdbc4.jdbc.SybDriver
java.lang.NoClassDefFoundError: Could not initialize class oracle.jdbc.OracleDriver
        at java.base/java.lang.Class.forName0(Native Method)
        at java.base/java.lang.Class.forName(Class.java:398)
        at java.sql/java.sql.DriverManager.isDriverAllowed(DriverManager.java:555)
        at java.sql/java.sql.DriverManager.isDriverAllowed(DriverManager.java:547)
        at java.sql/java.sql.DriverManager.getDrivers(DriverManager.java:449)
        at java.sql/java.sql.DriverManager.getDrivers(DriverManager.java:426)
        at com.sybase.jdbc4.jdbc.SybDriver.registerWithDriverManager(Unknown Source)
        at com.sybase.jdbc4.jdbc.SybDriver.<init>(Unknown Source)
        at com.sybase.jdbc4.jdbc.SybDriver.<clinit>(Unknown Source)
        at java.base/java.lang.Class.forName0(Native Method)
        at java.base/java.lang.Class.forName(Class.java:315)
        ... my code

所以我可以看到我试图获取的驱动程序名称是 com.sybase.jdbc4.jdbc.SybDriver,这是正确的,但出于某种原因,DriverManager 正在寻找 oracle.jdbc.OracleDriver。

这是怎么回事?这段代码多年来一直运行良好,我唯一能想到的其他相关信息是我最近将这台机器上的 JDK 升级到了 Open JDK 11。

>java -version
openjdk version "11.0.2" 2019-01-15
OpenJDK Runtime Environment 18.9 (build 11.0.2+9)
OpenJDK 64-Bit Server VM 18.9 (build 11.0.2+9, mixed mode)

最佳答案

这不是一个完整的答案,但它似乎是一个与自动驱动程序加载相结合的类加载问题。

当您显式使用 Class.forName 加载 JDBC 驱动程序时,该驱动程序应向 java.sql.DriverManager 注册自己。

查看堆栈跟踪,特别是:

    at java.sql/java.sql.DriverManager.getDrivers(DriverManager.java:426)
    at com.sybase.jdbc4.jdbc.SybDriver.registerWithDriverManager(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybDriver.<init>(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybDriver.<clinit>(Unknown Source)
    at java.base/java.lang.Class.forName0(Native Method)

Sybase 驱动程序在 (?) 注册自身之前错误地检查当前注册的驱动程序(使用 DriverManager.getDrivers)。更糟糕的是,它是从驱动程序构造函数而不是静态初始化程序执行此操作的,这可能会导致驱动程序加载死锁。正确的行为驱动程序应该从 JDBC 4.3 第 9.2 节中指定的静态初始化器调用 DriverManager.registerDriver:

JDBC drivers must implement the Driver interface, and the implementation must contain a static initializer that will be called when the driver is loaded. This initializer registers a new instance of itself with the DriverManager, as shown in CODE EXAMPLE 9-1.

public class AcmeJdbcDriver implements java.sql.Driver {
    static {
        java.sql.DriverManager.registerDriver(new AcmeJdbcDriver());
    }
    ... 
} 

CODE EXAMPLE 9-1 Example static initializer for a driver implementing java.sql.Driver

When a Driver implementation is loaded, the static initializer will automatically register an instance of the driver.

因为 DriverManager.getDrivers 被调用,它会自动在 META-INF/service/java.sql.Driver 文件(以及那些在系统属性 jdbc.drivers)。

看起来 Oracle JDBC 驱动程序是通过这种方式加载的,但是随后在 isDriverAllowed 中检查驱动程序是否在当前类加载器中可用失败并显示NoClassDefFoundError(检查捕获异常,但不捕获错误,也许它应该)。

作为解决方法,您应该从类路径中删除 Oracle JDBC 驱动程序,或者查明它在当前类加载器中不可用的原因。

作为进一步诊断,尝试调用 DriverManager.getDrivers()Class.forName("oracle.jdbc.Driver) 甚至 new oracle.jdbc .Driver() 在你的代码中,看看会发生什么。

您可能还想检查 Sybase 驱动程序的版本,如果有更新的版本则不会执行此检查,尽管这可能只会导致代码中其他地方出现错误。

关于java - 获取 java.lang.NoClassDefFoundError : Could not initialize class oracle. jdbc.OracleDriver 异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54960627/

相关文章:

java - 为字符串中的每个字符分配一组新字符

java - 在 Packages exec 期间不允许进行 DML 操作

database - 如何为 Oracle、Informix 和 Sybase 生成 2TB+ 大小的数据

java - 类加载器在加载资源包中的意义

java - 正弦函数在 Java 中如何工作?

java - 当我向变量分配函数时,如何让变量自行更新

java - ResultSet 不是用于INSERT 查询?错误消息 : Type mismatch: cannot convert from int to String

sybase - 在 sybase 中获取上次索引重建日期

java - 使用 MyBatis Sql 查询在 Spring Boot 应用程序中选择别名列

java - 如何使用spring boot格式化json日期格式