Class.forName("com.mysql.jdbc.Driver")理解

场景:
    为了理解操作jdbc时需要先使用Class.forName("com.mysql.jdbc.Driver")这行代码的作用。
解析:
    1.使用Class.forName("com.mysql.jdbc.Driver")后,jvm会加载这个类
    2.加载这个类后,进入com.mysql.jdbc.Driver,会立即注册:

    
	// Register ourselves with the DriverManager
	static {
		try {
			java.sql.DriverManager.registerDriver(new Driver());
		} catch (SQLException E) {
			throw new RuntimeException("Can't register driver!");
		}
	}

3.进入java.sql.DriverManager,DriverManager管理器会调用注册方法,并把Driver放入registeredDrivers列表中:

 public static synchronized void registerDriver(java.sql.Driver driver)
        throws SQLException {

        /* Register the driver if it has not already been added to our list */
        if(driver != null) {
            registeredDrivers.addIfAbsent(new DriverInfo(driver));
        } else {
            // This is for compatibility with the original DriverManager
            throw new NullPointerException();
        }

        println("registerDriver: " + driver);

    }

 以上代码,registeredDrivers是如下类型

private final static CopyOnWriteArrayList<DriverInfo> registeredDrivers = new CopyOnWriteArrayList<DriverInfo>();

4.在自己代码中,调用DriverManager.getConnection获取连接时,如下:

    @CallerSensitive
    public static Connection getConnection(String url,
        String user, String password) throws SQLException {
        java.util.Properties info = new java.util.Properties();

        if (user != null) {
            info.put("user", user);
        }
        if (password != null) {
            info.put("password", password);
        }

        return (getConnection(url, info, Reflection.getCallerClass()));
    }

   5.进入内部的getConnection(url, info, Reflection.getCallerClass())

private static Connection getConnection(
        String url, java.util.Properties info, Class<?> caller) throws SQLException {
        /*
         * When callerCl is null, we should check the application's
         * (which is invoking this class indirectly)
         * classloader, so that the JDBC driver class outside rt.jar
         * can be loaded from here.
         */
        ClassLoader callerCL = caller != null ? caller.getClassLoader() : null;
        synchronized (DriverManager.class) {
            // synchronize loading of the correct classloader.
            if (callerCL == null) {
                callerCL = Thread.currentThread().getContextClassLoader();
            }
        }

        if(url == null) {
            throw new SQLException("The url cannot be null", "08001");
        }

        println("DriverManager.getConnection(\"" + url + "\")");

        // Walk through the loaded registeredDrivers attempting to make a connection.
        // Remember the first exception that gets raised so we can reraise it.
        SQLException reason = null;

        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;
                    }
                }

            } else {
                println("    skipping: " + aDriver.getClass().getName());
            }

        }

        // if we got here nobody could connect.
        if (reason != null)    {
            println("getConnection failed: " + reason);
            throw reason;
        }

        println("getConnection: no suitable driver found for "+ url);
        throw new SQLException("No suitable driver found for "+ url, "08001");
    }
}

6.以上代码会去遍历一个列表,registeredDrivers列表里即是注册过的所有驱动
7.在getConnection(url, info, Reflection.getCallerClass())内部遍历registeredDrivers, 取出每个驱动进行判断即进入isDriverAllowed(aDriver.driver, callerCL)判断

private static boolean isDriverAllowed(Driver driver, ClassLoader classLoader) {
        boolean result = false;
        if(driver != null) {
            Class<?> aClass = null;
            try {
                aClass =  Class.forName(driver.getClass().getName(), true, classLoader);
            } catch (Exception ex) {
                result = false;
            }

             result = ( aClass == driver.getClass() ) ? true : false;
        }

        return result;
    }

 以上,所有注册驱动是否都一一匹配,都有即返回true.
8.以上都为true的话,依次执行如下代码
      Connection con = aDriver.driver.connect(url, info);
      如果连接成功,则返回con,没成功就继续遍历
 9.找不到不为null的Connection时,就抛出找不到合适驱动
      即可No suitable driver found ......

结论:只有com.mysql.jdbc.Driver提前被加载,在驱动管理器DriverManager获取连接时候才能正常验证通过,并创建连接.

以上,TKS.

<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文件。 所以,用户的问题可能涉及到不同版本的JDBCMySQL驱动的差异。需要分情况讨论:旧版本需要显式加载,而新版本可以自动加载。但有时候即使使用新版本,可能需要手动加载以确保兼容性,比如引用[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、付费专栏及课程。

余额充值