Class.forName() vs Class.loadClass()

ClassLoader.loadClass()与Class.forName()都是通过反射来构造类的方法。但是他们的用法还是有一定区别的。
在讲区别之前,先把类的加载过程整理一下。

在Java中,类装载器把一个类装入Java虚拟机中,要经过三个步骤来完成:装载、链接和初始化,其中链接又可以分成校验、准备和解析三步,除了解析外,其它步骤是严格按照顺序完成的,各个步骤的主要工作如下:

装载:查找和导入类或接口的二进制数据;

链接:执行下面的校验、准备和解析步骤,其中解析步骤是可以选择的;

校验:检查导入类或接口的二进制数据的正确性;

准备:给类的静态变量分配并初始化存储空间;

解析:将符号引用转成直接引用;

初始化:激活类的静态变量的初始化Java代码和静态Java代码块。

Class.forName(className)方法,
其实调用的方法是 Class.forName(className,true,classloader);
注意看第2个boolean参数,它表示的意思,在loadClass后必须初始化。
比较下前面准备jvm加载类的知识,可以清晰的看到在执行过此方法后,目标对象的 static块代码已经被执行,static参数也已经被初始化。

ClassLoader.loadClass(className)方法,
其实调用的方法是 ClassLoader.loadClass(className,false);
注意看第2个 boolean参数,该参数表示目标对象被装载后不进行链接,这就意味这不会去执行该类静态块中间的内容。

因此2者的区别就显而易见了。

一般情况下,这两个方法效果一样,都能装载Class。但如果程序依赖于Class是否被初始化,就必须用Class.forName(name)了。

例如,在JDBC编程中,常看到这样的用法,Class.forName("com.mysql.jdbc.Driver"),如果换成了 getClass().getClassLoader().loadClass("com.mysql.jdbc.Driver"),就不行。
为什么呢?打开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!");
    }
}

Driver在static块中会注册自己到java.sql.DriverManager。而static块就是在Class的初始化中被执行。所以这个地方就只能用Class.forName(className)。

时间: 2024-10-29 17:35:04

Class.forName() vs Class.loadClass()的相关文章

Class.forName和ClassLoader.loadClass的比较

Class的装载分了三个阶段,loading(加载),linking(链接)和initializing(初始化),分别定义在The Java Language Specification的12.2,12.3和12.4. Class.forName(className)实际上是调用Class.forName(className, true, this.getClass().getClassLoader()).注意第二个参数,是指Class被loading后是不是必须被初始化.  ClassLoad

理解class.forName()(good--字节码层面)

使用jdbc方式连接数据库时会使用一句代码Class.forName(String className).这句话是什么意思呢?首先说一点Class.forName(String className)这个方法的作用是装载className这个字符串指定的类. 官方文档   返回与带有给定字符串名的类或接口相关联的 Class 对象.调用此方法等效于:   Class.forName(className, true, currentLoader)  其中 currentLoader 表示此类的定义类

jdbc中class.forname的作用_java

使用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 = "";   Connectio

Apache Spark源码走读(七)Standalone部署方式分析&sql的解析与执行

<一>Standalone部署方式分析 楔子 在Spark源码走读系列之2中曾经提到Spark能以Standalone的方式来运行cluster,但没有对Application的提交与具体运行流程做详细的分析,本文就这些问题做一个比较详细的分析,并且对在standalone模式下如何实现HA进行讲解. 没有HA的Standalone运行模式 先从比较简单的说起,所谓的没有ha是指master节点没有ha. 组成cluster的两大元素即Master和Worker.slave worker可以有

HotSpot运行时概览#1

原文地址:http://openjdk.java.net/groups/hotspot/docs/RuntimeOverview.html 命令行参数处理 有许多的命令行选项和环境变量可以影响到HotSpot虚拟机的性能.其中有些选项直接由启动器处理(例如-server和-client),有些则是启动器先加工一下再交给虚拟机处理,但大部分选项还是由启动器直接交给虚拟机来处理. 主要有三类选项:标准选项,非标准选项,开发者选项.所有的JVM实现都要支持标准选项,即使不同的版本也要稳定支持(不管选项

java反射机制剖析(一)—简介

    由之前动态代理的学习再次接触到反射这个知识点,第二次接触了所以做了一些稍微深入的了解.那么,对于反射这部分的内容我打算分三篇博客来总结.本篇博客先对反射做一个大概的了解,包括反射有关的RTTI.定义的理解以及涉及到的其他知识的简介. 回顾     java之前我接触反射这个知识,是在大话设计中的抽象工厂模式里,通过反射+配置文件来优化抽象工厂提高其应对需求变更的灵活性.当时对于反射的认知仅仅是它是一种技术,一种实例化对象的技术,一种实例化对象不依赖于写死的代码的技术.简单的说就是,它是一

jvm系列(一):java类的加载机制

java类的加载机制   1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构.类的加载的最终产品是位于堆区中的Class对象,Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口. 类加载器并不需要等到某个类被"首次主动使用"时再加载它,JVM规范允许类加载器在预料某个类将要被使用时就预先加载

使用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 = DriverManag

一个用Class.forName()做的动态装载小东东,有助于理解Class.forName()的应用

动态 一直都在连JDBC中用到Class.forName(),当用到XML动态解析分派的时候遇到了Class名字符串无法作为Class名实例化的问题,寻寻觅觅终于找到一个解决的办法:public class DynamicLoader{    public static void main(String args[]) throws Exception{        Class toRun = Class.forName(args[0]);        String[] newArgs=sc