JDBC在getConnection之前为什么要调用Class.forName

本文深入解析了Class.forName(driver)方法的工作原理,包括如何将驱动类加载到内存,以及为什么在尝试使用数据库连接时必须调用此方法。通过分析不同数据库驱动的静态初始化块,揭示了注册驱动的机制,并提供了使用DriverManager.registerDriver()作为替代方案的示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

获取一个数据库连接的通用模板如下:

String driver = "oracle.jdbc.OracleDriver";
String url = "jdbc:oracle:thin:@127.0.0.1:1521:orcl";
String user = "scott";
String password = "ticmy";
Class.forName(driver);
Connection conn = DriverManager.getConnection(url, user, password);

里面有个Class.forName(driver),这句话有什么作用?将驱动类load到内存?如果没有这句会怎么样?运行发现,如果去掉这一句会有以下异常:
java.sql.SQLException: No suitable driver found for xxx….

在解释具体原因之前先简单看下Class.forName做了什么。

假设一个类以前从来没有被装进内存过,Class.forName(String className)这个方法会做以下几件事情:
1、装载。将字节码读入内存,并产生一个与之对应的java.lang.Class类对象
2、连接。这一步会验证字节码,为static变量分配内存,并赋默认值(0或null),并可选的解析符号引用(这里不理解没关系)
3、初始化。为类的static变量赋初始值,假如有static int a = 1;这个将a赋值为1的操作就是这个时候做的。除此之外,还要调用类的static块。(这一步是要点)

Class.forName(String className)方法会将这三步都做掉,如下面的例子:

package com.ticmy.oracle;
 
public class TestClinit {
    public static void main(String[] args) throws Exception {
        Class.forName("com.ticmy.oracle.ABC");
    }
}
class ABC {
    private static int a = getNum();
    static {
        System.out.println("this is static block");
    }
    public static int getNum() {
        System.out.println("getNum");
        return 1;
    }
}

程序的运行结果是:
getNum
this is static block

那么,Class.forName(driver)这个driver类里有没有什么static块呢?去探究一下。例子用的是Oracle,反编译下oracle.jdbc.OracleDriver,发现其继承了oracle.jdbc.driver.OracleDriver,那么继续看这个oracle.jdbc.driver.OracleDriver,确实有个static块,里面有这样的代码:

static {
    Timestamp localTimestamp = Timestamp.valueOf("2000-01-01 00:00:00.0");
    try {
      if (defaultDriver == null) {
        defaultDriver = new OracleDriver();
        DriverManager.registerDriver(defaultDriver);
      }
    } catch (RuntimeException localRuntimeException) {
    } catch (SQLException localSQLException){}
    _Copyright_2004_Oracle_All_Rights_Reserved_ = null;
}

再看看mysql吧:com.mysql.jdbc.Driver:
同样发现了static块,里面代码如下:

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

再看一个db2:com.ibm.db2.jcc.DB2Driver:
也发现了static块:

static {
    if (o.Nb != null) {
      exceptionsOnLoadDriver__ = dg.a(o.Nb, exceptionsOnLoadDriver__);
    }
 
    try {
      registeredDriver__ = new DB2Driver();
      DriverManager.registerDriver(registeredDriver__);
    }
    catch (SQLException localSQLException) {
      exceptionsOnLoadDriver__ = new SqlException(null,
      "Error occurred while trying to register Jcc driver with JDBC 1 Driver Manager");
      exceptionsOnLoadDriver__.setNextException(localSQLException);
    }
}

无一例外地,发现里面都有DriverManager.registerDriver(driver)的调用。那么是不是可以将开头的例子中的Class.forName换成DriverManager.registerDriver呢?

String url = "jdbc:oracle:thin:@127.0.0.1:1521:orcl";
String user = "scott";
String password = "ticmy";
DriverManager.registerDriver(new OracleDriver());
Connection conn = DriverManager.getConnection(url, user, password);
System.out.println(conn);
conn.close();

经过测试发现OK。现在,已经知道Class.forName(driver)的根本目的就是为了调用DriverManager.registerDriver

Class.forName还有个重载的方法:Class.forName(String name, boolean initialize, ClassLoader loader),Class.forName(String className)就等价于Class.forName(className, true, currentLoader),注意中间的参数为true,这个参数的含义就是要不要初始化。如果此参数为true且指定的类以前没有被初始化过,就会去初始化。

另外,jdbc4已经不需要显式的调用Class.forName了,在jdbc4中,调用getConnection的时候DriverManager会自动去加载合适的驱动。

<think>好的,我需要帮助用户解决如何正确使用Class.forName加载MySQL驱动的问题。首先,我得回顾一下用户提供的引用资料,确保答案准确。 根据引用[1],Class.forName("com.mysql.jdbc.Driver")的作用是让JVM加载这个类,然后执行静态代码块注册驱动到DriverManager。引用[3]提到,加载Driver类时会执行静态块中的registerDriver方法。但引用[4]指出,JDBC 4.0及以上版本支持自动加载驱动,不需要显式调用Class.forName,只要驱动jar包里有META-INF/services/java.sql.Driver文件。 所以,用户的问题可能涉及到不同版本的JDBC和MySQL驱动的差异。需要分情况讨论:旧版本需要显式加载,而新版本可以自动加载。但有时候即使使用新版本,可能需要手动加载以确保兼容性,比如引用[2]中的代码示例用了newInstance,虽然可能不必要,但用户需要注意这一点。 还要注意包名是否正确,比如引用[3]提到com.mysql.cj.jdbc.Driver,可能用户使用的驱动版本不同,包路径有变化。此外,引用[5]提到硬编码的问题,建议使用配置文件来管理驱动类名,提高灵活性。 接下来需要组织答案,分步骤说明传统方式和现代方式,注意事项和常见错误。最后生成相关问题,覆盖版本差异、自动加载原理、配置方法等。</think>### 如何正确使用 Class.forName 加载 MySQL 驱动? #### **步骤解析** 1. **传统手动加载方式(JDBC 4.0 以前)** 在 JDBC 4.0 之前,必须显式调用 `Class.forName("com.mysql.jdbc.Driver")` 加载驱动。 - **作用**:触发 JVM 加载 `Driver` 类,执行其静态代码块中的 `DriverManager.registerDriver(new Driver())`,将驱动注册到 JDBC 的 `DriverManager`[^1][^3]。 - **代码示例**: ```java Class.forName("com.mysql.jdbc.Driver"); // 旧版 MySQL 驱动 Connection conn = DriverManager.getConnection(url, user, password); ``` 2. **现代自动加载方式(JDBC 4.0 及以上)** JDBC 4.0 引入了 SPI(服务提供者接口)机制,**无需手动调用** `Class.forName`。只要满足以下条件: - MySQL 驱动版本 ≥ 5.1.6(包含 `META-INF/services/java.sql.Driver` 文件)[^4]; - JDK 版本 ≥ 1.6(支持 SPI 机制)。 - **代码示例**: ```java // 直接获取连接即可 Connection conn = DriverManager.getConnection(url, user, password); ``` 3. **注意事项** - **驱动类名差异**:MySQL 8.0+ 使用 `com.mysql.cj.jdbc.Driver`,旧版用 `com.mysql.jdbc.Driver`。 - **兼容性处理**:若不确定环境是否支持自动加载,可保留 `Class.forName` 调用(需捕获 `ClassNotFoundException`)[^5]。 - **避免硬编码**:建议将驱动类名配置在外部文件(如 `properties` 文件),通过读取配置加载,提高灵活性[^5]。 --- #### **常见错误与解决** 1. **`ClassNotFoundException`** - **原因**:驱动类名错误或未导入驱动 JAR 包。 - **解决**:检查包路径是否与驱动版本匹配(如 MySQL 8.0+ 使用 `com.mysql.cj.jdbc.Driver`)。 2. **`No suitable driver found`** - **原因**:未正确注册驱动(旧版本未调用 `Class.forName` 或自动加载失败)。 - **解决**:显式调用 `Class.forName` 或确认驱动 JAR 包含 `META-INF/services/java.sql.Driver` 文件。 --- #### **代码示例对比** ```java // 传统方式(显式加载) try { Class.forName("com.mysql.cj.jdbc.Driver"); // MySQL 8.0+ Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "user", "password"); } catch (ClassNotFoundException | SQLException e) { e.printStackTrace(); } // 现代方式(自动加载) try { Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "user", "password"); } catch (SQLException e) { e.printStackTrace(); } ``` ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值