解决JDBC的class.forName()问题

 更新时间:2022年09月27日 09:23:30   作者:蓝黑2020  
这篇文章主要介绍了关于JDBC的class.forName()问题,比较两个Java文件可见,连接Db2和连接MySQL的方式非常类似,唯一的区别在于,调用 DriverManager.getConnection() 方法时,传入的URL不同,本文给大家详细讲解,需要的朋友参考下

环境

  • Ubuntu 22.04
  • IntelliJ IDEA 2022.1.3
  • JDK 17.0.3
  • Db2 v11.5.0.0
  • MySQL Ver 8.0.30

准备

Db2

在Db2的 sample 数据库中,创建表 t1 ,并插入一些数据。如下:

➜  ~ db2 "select * from t1"

C1          C2          C3         
----------- ----------- -----------
          1         444           -
          2         222           -
          3         333           -

  3 record(s) selected.

MySQL

在MySQL的 repo 数据库中,创建表 t1 ,并插入一些数据。如下:

mysql> select * from t1;
+------+-------+
| c1   | c2    |
+------+-------+
|    1 |  9800 |
|    2 | 10200 |
+------+-------+
2 rows in set (0.00 sec)

代码

创建Maven项目 test0924

修改 pom.xml 文件,添加依赖:

......
        <!-- https://mvnrepository.com/artifact/com.ibm.db2/jcc -->
        <dependency>
            <groupId>com.ibm.db2</groupId>
            <artifactId>jcc</artifactId>
            <version>11.5.7.0</version>
        </dependency>

        <!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.29</version>
        </dependency>
......

如上,在项目添加了Db2和MySQL的JDBC驱动。

Db2

创建类 Test0924_Db2

package pkg1;

import java.sql.*;

public class Test0924_Db2 {
    public static void main(String[] args) throws ClassNotFoundException {
//        Class.forName("com.ibm.db2.jcc.DB2Driver");
        try (
                Connection connection = DriverManager.getConnection("jdbc:db2://localhost:50000/sample",
                        "db2inst1", "passw0rd");

                Statement stmt = connection.createStatement();

                ResultSet rs = stmt.executeQuery("select * from t1");
        ) {
//            System.out.println(connection.getClass().getName());

            while (rs.next()) {
                System.out.println(rs.getInt(1));
            }
        } catch (SQLException e) {
            throw new RuntimeException(e);
        }
    }
}

运行程序,结果如下:

1
2
3

MySQL

创建类 Test0924_Mysql

package pkg1;

import java.sql.*;

public class Test0924_Mysql {
    public static void main(String[] args) throws ClassNotFoundException {
//        Class.forName("com.mysql.jdbc.Driver");
        try (
                Connection connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/repo",
                        "root", "123456");

                Statement stmt = connection.createStatement();

                ResultSet rs = stmt.executeQuery("select * from t1");
        ) {
//            System.out.println(connection.getClass().getName());

            while (rs.next()) {
                System.out.println(rs.getInt(1));
            }
        } catch (SQLException e) {
            throw new RuntimeException(e);
        }
    }
}

运行程序,结果如下:

1
2

分析

JDBC

比较两个Java文件可见,连接Db2和连接MySQL的方式非常类似,唯一的区别在于,调用 DriverManager.getConnection() 方法时,传入的URL不同:

  • Db2: jdbc:db2://localhost:50000/sample
  • MySQL: jdbc:mysql://localhost:3306/repo

更确切的说,只是协议的不同: db2 VS. mysql

我们知道,JDBC是一套标准,各个厂商分别有着自己的实现,也就是各自的JDBC驱动。这也就是为什么一开始,我们就先引入Db2和MySQL的JDBC驱动。

JDBC中几个重要的类:

  • java.sql.DriverManager
  • java.sql.Connection
  • java.sql.Statement
  • java.sql.ResultSet

注意: ConnectionStatementResultSet 都是需要关闭的,一种方法是在 finally 块里显式调用其 close() 方法。本例中,使用了Java 7引入的 try() 块来自动释放资源(它们都实现了 AutoCloseable 接口)。

class.forName()

以前我们学习JDBC的时候,被告知第一步要先使用 Class.forName() 方法,导入特定的JDBC驱动。

但是通过本文的两个例子,我们看到,即使省略这一步,也没有问题,DriverManager能够自动找到合适的驱动。

那么问题来了:

  • 调用 Class.forName() 方法,到底干了什么?
  • 为什么本文中不调用该方法也没问题?

我们知道,如果某个类之前没有被使用过,则调用 Class.forName() 方法,会做几件事情,包括实例化该类的Class对象,并且执行其static块,等等。

对于JDBC驱动,以Db2驱动为例,查看 com.ibm.db2.jcc.DB2Driver 类,可以找到如下代码:

    static {
        DB2BaseDataSource.class.getClass();

        try {
            registeredDriver__ = new DB2Driver();
            DriverManager.registerDriver(registeredDriver__);
        } catch (SQLException var1) {
            ap.f = lr.a(b7.a(DB2Driver.class, (ds)null, ErrorKey.ERROR_REGISTER_WITH_DRIVER_MGR, "10032"), ap.f);
        }
    }

可见,调用了 DriverManager.registerDriver() 方法注册了Db2的驱动。

同理,对于MySQL,它的驱动类 com.mysql.cj.jdbc.Driver (是 com.mysql.jdbc.Driver 类的父类)里有如下代码:

    static {
        try {
            DriverManager.registerDriver(new Driver());
        } catch (SQLException var1) {
            throw new RuntimeException("Can't register driver!");
        }
    }

可见,类似的,也是调用了 DriverManager.registerDriver() 方法注册了MySQL的驱动。

由此,我们知道,调用 class.forName() 方法来装载驱动,其作用是注册了该驱动。

那么为什么本文中不调用方法也没问题呢?

java.sql.Connection 是一个接口,我们通过打印 connection.getClass().getName() 来看看具体的类名(参见代码中的注释部分)。

  • Db2: com.ibm.db2.jcc.t4.b
  • MySQL: com.mysql.cj.jdbc.ConnectionImpl

可见,即使不通过 class.forName() 方法来显式注册指定的驱动,而直接调用 DriverManager.getConnection() 方法,则根据传入的URL不同,也能获取正确的数据库连接。

可以去查看DriverManager的源码,大致如下:

......
        for (DriverInfo aDriver : registeredDrivers) {
            // If the caller does not have permission to load the driver then
            // skip it.
            if (isDriverAllowed(aDriver.driver, callerCL)) {
                try {
                    println("    trying " + aDriver.driver.getClass().getName());
                    Connection con = aDriver.driver.connect(url, info);
                    if (con != null) {
                        // Success!
                        println("getConnection returning " + aDriver.driver.getClass().getName());
                        return (con);
                    }
                } catch (SQLException ex) {
                    if (reason == null) {
                        reason = ex;
                    }
                }
......

也就是说,它会先生成驱动的列表,然后遍历列表,根据传入的URL,尝试使用当前驱动来连接数据库,如果能连上,就OK,否则就尝试下一个驱动。

当然,如果调用 Class.forName() 方法显式注册驱动,则会把驱动类放到列表的第一个,优先使用它来连接数据库。

到此这篇关于关于JDBC的class.forName()问题的文章就介绍到这了,更多相关JDBC的class.forName()内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • MyBatis 如何写配置文件和简单使用

    MyBatis 如何写配置文件和简单使用

    这篇文章主要介绍了MyBatis 如何写配置文件和简单使用的相关资料,需要的朋友可以参考下
    2017-01-01
  • linux二进制通用包安装mysql5.6.20教程

    linux二进制通用包安装mysql5.6.20教程

    这篇文章主要为大家详细介绍了linux二进制通用包安装mysql5.6.20的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-01-01
  • mysql 多个字段实现逗号拼接

    mysql 多个字段实现逗号拼接

    在MySQL数据库中,有时候我们需要将多个字段的值连接在一起,本文主要介绍了mysql 多个字段实现逗号拼接,具有一定的参考价值,感兴趣的可以了解一下
    2024-01-01
  • MySQL索引失效的典型案例

    MySQL索引失效的典型案例

    索引在我们使用MySQL数据库时可以极大的提高查询效率,然而,有时候因为使用上的一些瑕疵就会导致索引的失效,无法达到我们使用索引的预期效果,今天介绍一种MySQL中几种常见的索引失效的原因,可以在以后的工作中尽可能避免因索引失效带来的坑。
    2021-06-06
  • mysql对binlog的处理说明

    mysql对binlog的处理说明

    Mysql和其它开源数据库相比,具有更好的扩展性。其主要原因是它提供了存储引擎的开放接口。喜欢自己折腾数据库的程序员可以从这个接口起步,打造有个性的数据库。
    2011-07-07
  • mysql数据库优化必会的几个参数中文解释

    mysql数据库优化必会的几个参数中文解释

    对于自己配置mysql数据库的朋友,需要注意的几点,下面都是英文的解释,比较易懂方便和我一样需要优化配置mysql的朋友
    2008-09-09
  • 分析一个MySQL的异常查询的案例

    分析一个MySQL的异常查询的案例

    这篇文章主要介绍了分析一个MySQL的异常查询的案例,主要是针对索引方面的操作问题,需要的朋友可以参考下
    2015-05-05
  • MySQL执行update语句和原数据相同会再次执行吗

    MySQL执行update语句和原数据相同会再次执行吗

    这篇文章主要给大家介绍了关于MySQL执行update语句和原数据相同是否会再次执行的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用MySQL具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-04-04
  • 解决mysql时区问题导致错误Incorrect datetime value: '1970-01-01 00:00:01'

    解决mysql时区问题导致错误Incorrect datetime value: &apo

    这篇文章主要介绍了解决mysql时区问题导致错误Incorrect datetime value: '1970-01-01 00:00:01',具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-10-10
  • mac 装5.6版本mysql 设置密码的简易方法

    mac 装5.6版本mysql 设置密码的简易方法

    这篇文章主要介绍了mac 装5.6版本mysql 设置密码的简易方法,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2018-05-05

最新评论