Java知识记录(一)为什么Class.forName("com.mysql.jdbc.Driver") ?

初见问题

在学习注解的时候,我遇到了一个自定义注解模拟DBUtil的案例。

package util;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;

public class DBUtil {
    static String ip = "127.0.0.1";
    static int port = 3306;
    static String database = "test";
    static String encoding = "UTF-8";
    static String loginName = "root";
    static String password = "admin";
    static{
        try {
            Class.forName("com.mysql.jdbc.Driver");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
        public static Connection getConnection() throws SQLException {
        String url = String.format("jdbc:mysql://%s:%d/%s?characterEncoding=%s", ip, port, database, encoding);
      return DriverManager.getConnection(url, loginName, password);
}
        public static void main(String[] args) throws SQLException {
        System.out.println(getConnection());
    }
}   

其中Class.forName("com.mysql.jdbc.Driver"),这句话引起了我的注意。为什么这里为什么会平白无故的创建一个Class呢,而且为什么只是创建对象,而没有实例化呢?这个问题一下子带出了很多有意思的内容。

对象的创建

首先我们回顾一下创建类的两种方式:new以及反射。

通过new的方式创建一个类,这是我们最常见的一种方式。通过这种方式,我们可以调用任何是public的构造方法,它相对反射创建可以很高效的创建一个类。

而通过反射的方式,我们同样可以创建一个类。我们可以通过Class.forName()的方式获取一个类对象。然后通过将这个类newInstance()实例化,完成一个对象的创建。这里返回的对象为Object,所以我们通过需要做一个类型转化,来完成真正意义上的对象创建。还有另一种方式,就是通过ClassLoader.loadCLass()的方式来加载类,随后再实例化。

而问题的答案其实就存在于通过反射创建对象的过程当中。联系到JVM的知识,我们回忆一下,JVM是如何加载类的。

类的加载

首先我们要知道,JVM运行字节码需要将代码加载到内存中去。将class文件加载到内存中需要以下三个步骤:加载、链接、初始化。这个问题的核心在于初始化,我们不妨再大概回顾一下前两步的内容。

加载的意义在于查询字节流,并据此创建类。其中涉及到的知识点还有boot classloader启动类加载器,这玩意儿是C++写的,Java肯定访问不了的,但是我们必须有它才可以把一些顶级类加载器加载到JVM中,再进行将其他类加载进来。说到这里就又要提双亲委派模型了:低级类加载器应该先让它的上级类加载器查询后才确认是否干活。再需要说的就是类的唯一性了,我们可以借助不同的类加载器,来实现一个类的不同实现。这是由类加载器提供的命名空间实现的。

说完加载,链接的过程相对简单一点了。链接的主要作用就是把我们上一步中加载的类链接起来。它具体又有三个步骤:验证、准备、解析。但是解析阶段是非必须的。

最后回到我们问题的答案上,初始化阶段干了什么?给常量赋值以及执行静态代码块。我们看一下什么情况下会触发初始化。

  1. 当虚拟机启动时,初始化用户指定的主类;
  2. 当遇到用以新建目标类实例的 new 指令时,初始化 new 指令的目标类;
  3. 当遇到调用静态方法的指令时,初始化该静态方法所在的类;
  4. 当遇到访问静态字段的指令时,初始化该静态字段所在的类;
  5. 子类的初始化会触发父类的初始化;
  6. 如果一个接口定义了 default 方法,那么直接实现或者间接实现该接口的类的初始化,会触发该接口的初始化;
  7. 使用反射 API 对某个类进行反射调用时,初始化这个类;
  8. 当初次调用 MethodHandle 实例时,初始化该 MethodHandle 指向的方法所在的类。

通过上面的事例,我们大概可以猜测到之所以用Class.forName()的原因是在于要执行某些静态代码块或者为某些常量赋值。我们先解答一下为什么要用Class.forName(),而不是new和ClassLoader.loadClass()的问题。

  1. new开销太大,我们其实只需要执行某个静态代码块而已,并不需要将整个对象创建
  2. ClassLoader.loadClass()并没有将对象初始化,也就是没有执行我们需要的静态代码块

DriverManager

最后我们来揭晓一下这个神秘的静态代码块到底是什么。

public class Driver extends NonRegisteringDriver implements java.sql.Driver {
    public Driver() throws SQLException {
    }

    static {
        try {
            DriverManager.registerDriver(new Driver());//首先new一个Driver对象,并将它注册到DriverManage中
        } catch (SQLException var1) {
            throw new RuntimeException("Can't register driver!");
        }
    }
}

在JDBC规范中明确要求这个Driver类必须向DriverManager注册自己。之所以使用Class.forName("com.mysql.jdbc.Driver")的原因,就是为了执行这个静态代码块,使Driver在DriverManager中注册。

而在高版本的JDK中,我们已经不需要在手动的调用这种方法,在DriverManager的源码中有这么一个静态块。

/**
     * Load the initial JDBC drivers by checking the System property
     * jdbc.properties and then use the {@code ServiceLoader} mechanism
     */
    static {
        loadInitialDrivers();
        println("JDBC DriverManager initialized");
    }

让我们来探究一下loadInitialDrivers()的作用。

private static void loadInitialDrivers() {
        String drivers;
        try {
            drivers = AccessController.doPrivileged(new PrivilegedAction<String>() {
                public String run() {
                    return System.getProperty("jdbc.drivers");
                }
            });
        } catch (Exception ex) {
            drivers = null;
        }
        // If the driver is packaged as a Service Provider, load it.
        // Get all the drivers through the classloader
        // exposed as a java.sql.Driver.class service.
        // ServiceLoader.load() replaces the sun.misc.Providers()

        AccessController.doPrivileged(new PrivilegedAction<Void>() {
            public Void run() {

                ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
                Iterator<Driver> driversIterator = loadedDrivers.iterator();

                /* Load these drivers, so that they can be instantiated.
                 * It may be the case that the driver class may not be there
                 * i.e. there may be a packaged driver with the service class
                 * as implementation of java.sql.Driver but the actual class
                 * may be missing. In that case a java.util.ServiceConfigurationError
                 * will be thrown at runtime by the VM trying to locate
                 * and load the service.
                 *
                 * Adding a try catch block to catch those runtime errors
                 * if driver not available in classpath but it's
                 * packaged as service and that service is there in classpath.
                 */
                try{
                    while(driversIterator.hasNext()) {
                        driversIterator.next();
                    }
                } catch(Throwable t) {
                // Do nothing
                }
                return null;
            }
        });

重点是ServiceLoader.load(Driver.class)这行代码。

上面这行代码可以把类路径下所有jar包中META-INF/services/java.sql.Driver文件中定义的类加载上来,此类必须继承自java.sql.Driver。

最后我们看一下Iterator的next()方法做了什么就完全懂了,通过next()方法调用了:

private S nextService() {
    if (!hasNextService())
        throw new NoSuchElementException();
    String cn = nextName;
    nextName = null;
    Class<?> c = null;
    try {
        c = Class.forName(cn, false, loader); //看这里,Class.forName()
    } catch (ClassNotFoundException x) {
        fail(service,
             "Provider " + cn + " not found");
   }
    if (!service.isAssignableFrom(c)) {
       fail(service,
             "Provider " + cn  + " not a subtype");
    }
    try {
        S p = service.cast(c.newInstance());
        providers.put(cn, p);
        return p;
    } catch (Throwable x) {
        fail(service,
             "Provider " + cn + " could not be instantiated",
            x);
    }
    throw new Error();          // This cannot happen
}

说白了就是把原来我们手动的方式变成了框架自动的方式。这种方式也叫做SPI。关于SPI的知识我们可以自行去查询学习一下,也是挺有意思的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351