转载请标明出处:http://blog.csdn.net/xx326664162/article/details/51659663 文章出自:薛瑄的博客
你也可以查看我的其他同类文章,也会让你有一定的收货!
以下所示页码,是指《java编程思想》第四版的页码
#生成Class对象:
1.类名.class
说明:
- JVM将使用类装载器, 将类装入内存(前提是:类还没有装入内存),不做类的初始化工作。返回Class的对象(页码:P319)
- .class 只能返回类内field的默认值
2.Class.forName(“类名字符串”) (注:类名字符串是包名+类名)
说明:
- 装入类,并做类的静态初始化(初始化类中的静态方法和变量),返回Class的对象
- class.forName()只能返回类内field的默认值
3.实例对象.getClass() (页码:P316)
说明:
- 对类进行静态初始化、非静态初始化;
- getClass()是动态的,其余是静态的。
- getClass可以返回当前对象中field的最新值
- 返回引用运行时真正所指的对象(因为:子对象的引用可能会赋给父对象的引用变量中)所属类的Class对象
示例:
package ClassTest;
public class TestClass {
public static void main(String[] args) {
try {
// 测试.class
@SuppressWarnings("rawtypes")
Class testTypeClass = TestClassType.class;
System.out.println("testTypeClass---" + testTypeClass);
// 测试Class.forName()
@SuppressWarnings("rawtypes")
Class testTypeForName = Class.forName("ClassTest.TestClassType");
System.out.println("testTypeForName---" + testTypeForName);
// 测试Object.getClass()
TestClassType testTypeGetClass = new TestClassType();
System.out.println("testTypeGetClass---"
+ testTypeGetClass.getClass());
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
package ClassTest;
public class TestClassType {
// 构造函数
public TestClassType() {
System.out.println("----构造函数---");
}
// 静态代码块
static {
System.out.println("---静态的参数初始化---");
}
// 普通代码块
{
System.out.println("----非静态的参数初始化---");
}
}
运行结果如下:
testTypeClass---class ClassTest.TestClassType
---静态的参数初始化---
testTypeForName---class ClassTest.TestClassType
----非静态的参数初始化---
----构造函数---
testTypeGetClass---class ClassTest.TestClassType
- .class 初始化被延迟到了对静态方法或者非常数静态域(变量)进行首次引用时才执行。隐式的构造器是静态的
- Class.forName(“类名字符串”) 加载类的时候。静态方法和静态变量被初始化
- 普通代码块,是new类实例对象的时候执行。但在示例中,发现使用new示例时,并没有打印出“—静态的参数初始化—”。我的分析是,已经被加载的类,在实例化时,不会再执行静态初始化
因此,这段程序说明,三种方式生成Class对象,其实只有一个Class对象。在生成Class对象的时候,首先判断内存中是否已经加载。
生成Class对象的过程其实是如此的:
当我们编写一个新的java类时,JVM就会帮我们编译成class对象,存放在同名的.class文件中。在运行时,当需要生成这个类的对象,JVM就会检查此类是否已经装载内存中。若是没有装载,则把.class文件装入到内存中。若是装载,则根据class文件生成实例对象。
参考:
java反射中,Class.forName和classloader的区别(代码说话)
http://blog.csdn.net/zhangxichao100/article/details/51104971
http://blog.csdn.net/ozwarld/article/details/8277361