JDBC Class.forName作用(转)

 

 使用JDBC时,我们都会很自然得使用下列语句:

 

Class.forName("com.mysql.jdbc.Driver");    
String url = "jdbc:mysql://127.0.0.1/test?useUnicode=true&characterEncoding=utf-8";    
String user = "";    
String psw = "";    
Connection con = DriverManager.getConnection(url,user,psw);  
 


    为什么说很自然呢,因为无论是网上还是书本教程上得例子都是这样的,而且程序也确实正常运行了,于是大家也就心安理得的找葫芦画瓢下去了。

    一定要有这一句吗?不是的,我们完全可以用这样一句代替它:

 

com.mysql.jdbc.Driver driver = new com.mysql.jdbc.Driver();    
//or:    
//new com.mysql.jdbc.Driver();    
String url = "jdbc:mysql://127.0.0.1/test?useUnicode=true&characterEncoding=utf-8";    
String user = "";    
String psw = "";    
Connection con = DriverManager.getConnection(url,user,psw);    
 

 


    大家可能都看出个大概来了,我们只需要在调用DriverManager的getConnection方法之前,保证相应的Driver类已经被加载到jvm中,并且完成了类的初始化工作就行了,而具体是怎样实现这个功能却是没有讲究的。上面两种方法都可以实现这个功能,因此程序可以正常运行。注意了,如果我们进行如下操作,程序是不能正常运行的,因为这样仅仅使Driver类被装载到jvm中,却没有进行相应的初始化工作。

 

com.mysql.jdbc.Driver driver = null;    
//or:    
ClassLoader cl = new ClassLoader();    
cl.loadClass("com.mysql.jdbc.Driver");  
 

 

     我们都知道JDBC是使用Bridge模式进行设计的,DriverManager就是其中的Abstraction,java.sql.Driver是Implementor,com.mysql.jdbc.Driver是Implementor的一个具体实现(请参考GOF的Bridge模式的描述)。大家注意了,前一个Driver是一个接口,后者却是一个类,它实现了前面的Driver接口。

     Bridge模式中,Abstraction(DriverManager)是要拥有一个Implementor(Driver)的引用的,但是我们在使用过程中,并没有将Driver对象注册到DriverManager中去啊,这是怎么回事呢?jdk文档对Driver的描述中有这么一句:

     When a Driver class is loaded, it should create an instance of itself and register it with the DriverManager 

哦,原来是com.mysql.jdbc.Driver在装载完后自动帮我们完成了这一步骤。源代码是这样的:

 

package com.mysql.jdbc    
   
public class Driver extends NonRegisteringDriver implements java.sql.Driver {    
 // ~ Static fields/initializers    
 // --------------------------------------------- //    
 // Register ourselves with the DriverManager    
 //    
 static {    
    t ry {    
              java.sql.DriverManager.registerDriver(new Driver());    
          } catch (SQLException E) {    
              throw new RuntimeException("Can't register driver!");    
          }    
  }    
// ~ Constructors    
 // -----------------------------------------------------------    
/**   
  * Construct a new driver and register it with DriverManager   
  *    
  * @throws SQLException   
  *             if a database error occurs.   
  */   
 public Driver() throws SQLException {    
     // Required for Class.forName().newInstance()    
 }    
}

 可以看到DriveManager里面定义了静态方法,而mysql等驱动类都是内部用静态数据块的方式来初始化,完成Driver对象注册到DriverManager中,所以只要jvm装载了这些驱动类就会将Driver对象注册到DriverManager中去(Class.forName(xxx.xx.xx) 返回的是一个类;Class.forName(xxx.xx.xx);的作用是要求JVM查找并加载指定的类,也就是说JVM会执行该类的静态代码段 。静态代码是和class绑定的,class装载成功就表示执行了你的静态代码了。而且以后不会再走这段静态代码了。),后面就可以直接使用了

<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、付费专栏及课程。

余额充值