Java基础学习之:反射
1:通过Java代码运行各个阶段来理解反射
在学习反射以前,我们先来了解一下Java代码在计算机中所经历的三个阶段:
-
Source源代码阶段:.java被编译成*.class字节码文件。
-
Class类对象阶段:.class字节码文件被类加载器加载进内存,并将其封装成Class对象(用于在内存中描述字节码文件),Class对象将原字节码文件中的成员变量抽取出来封装成数组Field[],将原字节码文件中的构造函数抽取出来封装成数组Construction[],将成员方法封装成数组Method[]。当然Class类内不止这三个,还封装了很多,我们常用的就这三个。
-
RunTime运行时阶段:使用new创建对象的过程。
2:用以上三个阶段来区分获取Class对象的三种方式
-
【Source源代码阶段】 Class.forName(“全类名”):将字节码文件加载进内存,返回Class对象;
多用于配置文件,将类名定义在配置文件中,通过读取配置文件加载类。 -
【Class类对象阶段】 类名.class:通过类名的属性class获取;
多用于参数的传递 -
【Runtime运行时阶段】对象.getClass():此方法是定义在Objec类中的方法,因此所有的类都会继承此方法。
多用于对象获取字节码的方式代码部分:
public class Person { private int age; private String name; public Person() { } public Person(int age, String name) { this.age = age; this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } @Override public String toString() { return "Person{" + "age=" + age + ", name='" + name + '\'' + '}'; } } public class Mytest1 { public static void main(String[] args) throws ClassNotFoundException { // Class a=Class.forName("Person"); 一开始这么写报错了,要写全类名 Class a=Class.forName("com.ws.pojo.Person"); Class b= Person.class; Person p=new Person(); Class c=p.getClass(); System.out.println(a==b); //true System.out.println(b==c); //true } } /*这里面的abc都是Class类对象,都指向着person类。*/
3:java.lang.Class类的深入理解
-
Class类: Java程序在运行时,Java运行时系统一直对所有的对象进行所谓的运行时类型标识。这项信息纪录了每个对象所属的类。虚拟机通常使用运行时类型信息选准正确方法去执行,用来保存这些类型信息的类是Class类。
-
Class类对象:通过Class类创建的对象,每个Class类对象对应一个类。如上面代码的abc。
Java程序在运行时,Java运行时系统一直对所有的对象进行所谓的运行时类型标识。这项信息纪录了每个对象所属的类。虚拟机通常使用运行时类型信息选准正确方法去执行,用来保存这些类型信息的类是Class类。Class类封装一个对象和接口运行时的状态,当装载类时,Class类型的对象自动创建。(在上面的第二个阶段里面)
Class 没有公共构造方法。Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的,因此不能显式地声明一个Class对象。
虚拟机为每种类型管理一个独一无二的Class对象。也就是说,每个类(型)都有一个Class对象。运行程序时,Java虚拟机(JVM)首先检查是否所要加载的类对应的Class对象是否已经加载。如果没有加载,JVM就会根据类名查找.class文件,并将其Class对象载入。
基本的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也都对应一个 Class 对象。
每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共享该 Class 对象。
一般某个类的Class对象被载入内存,它就用来创建这个类的所有对象。
个人理解:上面的第二个步骤中(Class类对象阶段),类加载器将.class字节码文件装进内存,并将其封装成Class对象(用于在内存中描述字节码文件)====每一个.class字节码文件对应一个类,在将每一个.class文件加载进内存里时,jvm都会自动调用类加载器中的defineClass方法来创建Class对象(即每个类能够对应一个Class对象)。
4:自己的一个误区
s字节码文件对应一个类,在将每一个.class文件加载进内存里时,jvm都会自动调用类加载器中的defineClass方法来创建Class对象(即每个类能够对应一个Class对象)。
4:自己的一个误区
反射反射之所以为反射,不是类A的对象来反射到类A自己。而是Class类的一个对象来反射到类A。这是我自己一直以来的误区。