Java基础学习之:反射

本文详细解析Java反射机制,从代码运行阶段理解Class对象获取方式,探讨Class类的本质,纠正反射误解,并揭示Class类在运行时的角色。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Java基础学习之:反射

1:通过Java代码运行各个阶段来理解反射

在学习反射以前,我们先来了解一下Java代码在计算机中所经历的三个阶段:

  1. Source源代码阶段:.java被编译成*.class字节码文件。

  2. Class类对象阶段:.class字节码文件被类加载器加载进内存,并将其封装成Class对象(用于在内存中描述字节码文件),Class对象将原字节码文件中的成员变量抽取出来封装成数组Field[],将原字节码文件中的构造函数抽取出来封装成数组Construction[],将成员方法封装成数组Method[]。当然Class类内不止这三个,还封装了很多,我们常用的就这三个。

  3. RunTime运行时阶段:使用new创建对象的过程。

在这里插入图片描述

2:用以上三个阶段来区分获取Class对象的三种方式

  1. 【Source源代码阶段】 Class.forName(“全类名”):将字节码文件加载进内存,返回Class对象;
    多用于配置文件,将类名定义在配置文件中,通过读取配置文件加载类。

  2. 【Class类对象阶段】 类名.class:通过类名的属性class获取;
    多用于参数的传递

  3. 【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。这是我自己一直以来的误区。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值