转自:http://www.cnblogs.com/suibianle/p/6676215.html
为什么要把ClassLoader.loadClass(String name)和Class.forName(String name)进行比较呢,因为他们都能在运行时对任意一个类,都能够知道该类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性。
在比较它俩之前需先了解一下java类装载的过程
java类装载过程分为3步:
1:加载
Jvm把class文件字节码加载到内存中,并将这些静态数据装换成运行时数据区中方法区的类型数据,在运行时数据区堆中生成一个代表这个类
的java.lang.Class对象,作为方法区类数据的访问入口。
*释:方法区不仅仅是存放方法,它存放的是类的类型信息。
2:链接:执行下面的校验、准备和解析步骤,其中解析步骤是可选的
a:校验:检查加载的class文件的正确性和安全性
b:准备:为类变量分配存储空间并设置类变量初始值,类变量随类型信息存放在方法区中,生命周期很长,使用不当和容易造成内存泄漏。
*释:类变量就是static变量;初始值指的是类变量类型的默认值而不是实际要赋的值
c:解析:jvm将常量池内的符号引用转换为直接引用
3:初始化:执行类变量赋值和静态代码块
在了解了类装载过程之后我们继续比较二者区别:
其实该方法内部调用的是:Classloder. loadClass(name, false)
方法:Classloder. loadClass(String name, boolean resolve)
1:参数name代表类的全限定类名
2:参数resolve代表是否解析,resolve为true是解析该类
其实该方法内部调用的是:Class.forName(className, true, ClassLoader.getClassLoader(caller))
方法:Class.forName0(String name, boolean initialize, ClassLoader loader)
参数name代表全限定类名
参数initialize表示是否初始化该类,为true是初始化该类
参数loader 对应的类加载器
Class.forName得到的class是已经初始化完成的
Classloder.loaderClass得到的class是还没有链接的
有些情况是只需要知道这个类的存在而不需要初始化的情况使用Classloder.loaderClass,而有些时候又必须执行初始化就选择Class.forName
例如:数据库驱动加载就是使用Class.froName(“com.mysql.jdbc.Driver”),
下面我们来看看Driver的源代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public
class
Driver
extends
NonRegisteringDriver
implements
java.sql.Driver {
public
Driver()
throws
SQLException {
}
static
{
try
{
DriverManager.registerDriver(
new
Driver());
}
catch
(SQLException var1) {
throw
new
RuntimeException(
"Can\'t register driver!"
);
}
}
}
|
从Driver的源码中我们可以看出Driver这个类只有一个static块,这样我们需要初始化后才能得到DriverManager,所以我们选择使用Class.forName()