Java Class 类对象
java 中Class 类。在java 类库中,如下关系 Class
public final class Class
extends Object implements Serializable, GenericDeclaration, Type, AnnotatedElement
java.lang.Object
↳ java.lang.Class<T>
Class 类继承自Object 类,类 Class 的实例表示正在运行的 Java 应用程序中的类和接口。 枚举是一种类,注解是一种接口。 每个数组也属于一个类,该类反映为一个 Class 对象,该对象由具有相同元素类型和维数的所有数组共享。 原始 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象。
1、类没有公共构造函数。 相反,Java 虚拟机在加载类时以及通过调用类加载器中的 defineClass 方法时自动构造 Class 对象。Class类只存私有构造函数(无法通过new 创建Class类对象)
2、通常,我们在编写完java 代码时,编译器会自动为我们生成.class 文件(文件字节码)
3、运行程序时,Java虚拟机(JVM)首先检查是否所要加载的类对应的Class对象是否已经加载。如果没有加载,JVM就会根据类名查找.class文件,并将其Class对象载入,这是属于类加载的过程。多个实例对象会对应一个class 对象(类对象)。
4、Class类的对象作用是运行时提供或获得某个对象的类型信息。(反射机制就是运行时机制)
5、对于我们代码来说,什么时候会触发类加载(通过一个类的完全限定查找此类字节码文件,并利用字节码文件创建一个Class对象)呢?
答:使用时就会检查,此类是否已经加载了(懒加载机制),没加载那就调用类加载器进行加载。比如 new 一个实例对象。Class.formName(全类名)等。
java 中获取 类对象的方式
- 实例.getClass().getName 如下(摘自官网):
void printClassName(Object obj) {
System.out.println("The class of " + obj +
" is " + obj.getClass().getName());
}
- 类名.class.getName(摘自官网)
System.out.println("The name of class Foo is: "+Foo.class.getName());
- Class.formName(全类名方式)
Class clazz = Class.fromName("java.lang.String")
System.out.println(calzz.getName())
1、 Class.forName()
Return static Class<?> :forName(String className)
Returns the Class object associated with the class or interface with the given string name.
译:返回与具有给定字符串名称的类或接口关联的 Class 对象。
相当于如下 Java 提供的方法(下面这个方法可以动态控制类是否进行初始化,类的初始化是 另外一个阶段类加载阶段:
Class.forName(className, true, currentLoader)
引用官网对如上方法解释:
其中 currentLoader 表示当前类的定义类加载器。
例如,以下代码片段返回名为 java.lang.Thread 的类的运行时类描述符:
Class t = Class.forName("java.lang.Thread")
对 forName("X") 的调用导致名为 X 的类被初始化。
所以,Class.fromName(全类名) 生成了类对象,并且进行类的初始化操作
穿插:类加载过程阶段
加载:类加载过程的一个阶段:通过一个类的完全限定查找此类字节码文件,并利用字节码文件创建一个Class对象
链接:验证字节码的安全性和完整性,准备阶段正式为静态域分配存储空间,注意此时只是分配静态成员变量的存储空间,不包含实例成员变量,如果必要的话,解析这个类创建的对其他类的所有引用。
初始化:类加载最后阶段,若该类具有超类,则对其进行初始化,执行静态初始化器和静态初始化成员变量。
Class.froName 会抛出如下异常
注意:getClass方法和Class类的静态方法forName都将会触发类的初始化阶段,而字面常量获取Class对象的方式则不会触发初始化。
2、类对象的方法 newInstance()
public T newInstance ()
类对象.newInstance() 方法调用无参的构造函数 实例化对象。
官方解释:创建由此 Class 对象表示的类的新实例。 类被实例化,就好像由一个带有空参数列表的新表达式一样。 如果该类尚未初始化,则该类已初始化。
请注意,此方法会传播空构造函数抛出的任何异常,包括已检查的异常。 使用此方法可以有效地绕过编译器否则会执行的编译时异常检查。 Constructor.newInstance 方法通过将构造函数抛出的任何异常包装在(已检查的)InvocationTargetException 中来避免此问题。
3、new对象和newInstance的区别
new一个对象的时候,会先查看对象所属的类有没有被加载到内存,如果没有的话,就会先通过类的全限定名来加载。加载并初始化类完成后,再进行对象的创建工作。
如果是第一次使用该类,这样的话new一个对象就可以分为两个过程:加载并初始化类和创建对象。
newInstance 由2可知,此方法返回的是一个T 。所以用newInstance与用new是区别的,区别在于创建对象的方式不一样,前者是使用类加载机制完成的。newinstance() 方法通常适用于接口编程,如工厂方法。