我的编程空间,编程开发者的网络收藏夹
学习永远不晚

解决JDBC的class.forName()问题

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

解决JDBC的class.forName()问题

环境

  • 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()内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

解决JDBC的class.forName()问题

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Java使用Class.forName加载外部Jar类文件的问题怎么解决

本篇内容介绍了“Java使用Class.forName加载外部Jar类文件的问题怎么解决”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!故事背
2023-07-02

jdbc连接失效问题如何解决

要解决JDBC连接失效问题,可以尝试以下几个方法:检查数据库服务器是否正常运行:确保数据库服务器正常运行并且可以接收连接。可以通过尝试使用其他工具(如数据库管理工具)连接数据库来确认。检查网络连接:确保网络连接正常,包括数据库服务器和应用程
2023-10-24

解决maven不能下载oracle jdbc驱动的问题

目录maven不能下载oracle jdbc驱动问题解决方案maven配置Oracle JDBC driverwww.cppcns.com驱动1.查看本地oracle版本2.从本地安装的oracle目录中获取oracle 驱动3.将orac
2023-04-11

maven不能下载oracle jdbc驱动的问题怎么解决

本篇内容主要讲解“maven不能下载oracle jdbc驱动的问题怎么解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“maven不能下载oracle jdbc驱动的问题怎么解决”吧!mave
2023-07-06

Shard-Jdbc数据库扩容的场景和问题的解决方法

这篇文章主要介绍“Shard-Jdbc数据库扩容的场景和问题的解决方法”,在日常操作中,相信很多人在Shard-Jdbc数据库扩容的场景和问题的解决方法问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Shard
2023-06-02

jdbc-实现用户登录业务(解决sql注入问题)

package com.cqust;import java.sql.*;import java.util.HashMap;import java.util.Map;import java.util.Scanner;/**解决sql注入问题使用预编译在还没有拼接
jdbc-实现用户登录业务(解决sql注入问题)
2018-02-17

解决vue.jsnotdetected的问题

本文主要介绍了解决vue.jsnotdetected的问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-02-01

mysql中tinyint(1)对应java中的boolean问题及解决方案(jdbc查询)

首先展示下我的问题,可以看见id为66,64,63对应的month_age分别为1,1,0,但是数据库中这三条记录对应的month_age应该是11,5,0 造成这个错误的原因是我们在设计数据库时,将month_age这个字段的类型设为
2023-08-23

登录EasyConnect后无法通过jdbc访问服务器数据库问题的解决方法

描述一下近期使用EasyConnect遇到的问题,下面这篇文章主要给大家介绍了关于登录EasyConnect后无法通过jdbc访问服务器数据库问题的解决方法,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
2023-02-24

MySql的JDBC连接url地址的各项配置说明及所解决的问题

首先看一条jdbc的url地址: `url: jdbc:mysql://127.0.0.1:3306/test?useAffectedRows=true&autoReconnect=true&useUnicode=true&charact
2023-08-17

关于使用MyBatis简化JDBC开发和解决SQL语句警告的问题

这篇文章主要介绍了关于使用MyBatis简化JDBC开发和解决SQL语句警告的问题,如果idea和数据库没有建立链接,idea不识别表的信息,就会出现SQL语句的警告,需要的朋友可以参考下
2023-05-18

Python的问题解决: IOError

错误一:遇到一个很奇怪的问题, web.py代码里面报错IOError: [Errno 32] Broken pipe启动命令: nohup python xxx.py >> xxx.log &ssh登录到机器上, 启动, 不会出现远程ss
2023-01-31

pmm问题解决

今天pmm挂了,监控的界面为空,进行了如下解决: 1. 进入容器查看日志 docker ps -a docker exec -it e4916410b314 /bin/bash /var/log 下面各种日志 /var/log/grafana/
pmm问题解决
2015-11-23

编程热搜

目录