19

我需要测试与数据库的 JDBC 连接。执行此操作的 java 代码应该很简单:

DriverManager.getConnection("jdbc connection URL", "username", "password");

驱动程序管理器将为给定的连接 URL 查找适当的驱动程序。但是我需要能够在运行时加载 JDBC 驱动程序(jar)。即我在运行上述代码片段的java 应用程序的类路径上没有JDBC 驱动程序。

所以我可以使用此代码加载驱动程序,例如:

URLClassLoader classLoader = new URLClassLoader(new URL[]{"jar URL"}, this.getClass().getClassLoader());
Driver driver = (Driver) Class.forName("jdbc driver class name", true, classLoader).newInstance();

但是驱动程序管理器仍然不会接受它,因为我无法告诉它使用哪个类加载器。我尝试设置当前线程的上下文类加载器,但它仍然不起作用。

有人对实现这一目标的最佳方法有任何想法吗?

4

2 回答 2

20

来自文章在运行时选择您的 JDBC 驱动程序;我只是在这里发布代码以供参考。

这个想法是诱使驱动程序管理器认为驱动程序是从系统类加载器加载的。为此,我们使用这个类:

public class DelegatingDriver implements Driver
{
    private final Driver driver;

    public DelegatingDriver(Driver driver)
    {
        if (driver == null)
        {
            throw new IllegalArgumentException("Driver must not be null.");
        }
        this.driver = driver;
    }

    public Connection connect(String url, Properties info) throws SQLException
    {
       return driver.connect(url, info);
    }

    public boolean acceptsURL(String url) throws SQLException
    {
       return driver.acceptsURL(url);
    }

    public DriverPropertyInfo[] getPropertyInfo(String url, Properties info) throws SQLException
    {
        return driver.getPropertyInfo(url, info);
    }

    public int getMajorVersion()
    {
        return driver.getMajorVersion();
    }

    public int getMinorVersion()
    {
        return driver.getMinorVersion();
    }

    public boolean jdbcCompliant()
    { 
        return driver.jdbcCompliant();
    }
}

这样,您注册的驱动程序DelegatingDriver就是使用系统类加载器加载的类型。您现在只需使用您想要的任何类加载器加载您真正想要使用的驱动程序。例如:

URLClassLoader classLoader = new URLClassLoader(new URL[]{"path to my jdbc driver jar"}, this.getClass().getClassLoader());
Driver driver = (Driver) Class.forName("org.postgresql.Driver", true, classLoader).newInstance();
DriverManager.registerDriver(new DelegatingDriver(driver)); // register using the Delegating Driver

DriverManager.getDriver("jdbc:postgresql://host/db"); // checks that the driver is found
于 2008-11-14T01:06:37.660 回答
5

问题是DriverManager执行“使用直接调用者的类加载器实例的任务”。请参阅Java 编程语言的安全编码指南 2.0 版的指南 6-3 。在这种情况下,系统类加载器绝不是特殊的。

只是为了好玩,不久前我写了一篇关于这个主题的博客文章(编辑:现已失效)。我的解决方案虽然比Nick Sayer 的解决方案更复杂,但更完整,甚至可以使用不受信任的代码工作(编辑:不推荐使用不受信任代码的概念,以便在 Java 17 中删除)。另外 noteURLClassLoader.newInstance优先于new URLClassLoader.

于 2008-11-14T11:33:05.480 回答