反射、 Class类、JVM的类加载机制、Class的常用方法

news/2025/3/13 13:33:36/

DAY11.1 Java核心基础

反射

重点和难点,应用面很广

大部分库和框架都需要用到反射机制,它是动态语言的关键,但是概念抽象不好理解

反射:通过实例化类映射到类,从而获取类的信息

概括说就是:常规情况是通过类来创建实例化对象,而反射就是将这个过程反转,通过实例化对象获取类的信息

Class类(反射 API 的入口)

Class类是反射的基础

Class类:是专门用来描述其它类的类

对每一种对象,JVM 都会实例化一个 java.lang.Class 的实例,java.lang.Class 为我们提供了在运行时访问对象的属性和类型信息的能力。

Class 还提供了创建新的类和对象的能力。最重要的是,Class 是调用其他反射 API 的入口,我们必须先获得一个 Class 实例才可以进行接下来的操作。

创建Class对象的三种方式

  • forName():Class的静态方法forName()方法

  • .class:通过实体类的.class直接取得

  • getClass():通过创建实体类的getClass()方法获取

java">public static void main(String[] args) throws IOException, ClassNotFoundException, InstantiationException, IllegalAccessException {File file1 = new File("H:\\Java代码库\\DeepSeek_Tesat\\src\\main\\java\\com\\deepSeek\\shuwu\\Day10\\test3.txt");Class class1 = Class.forName("com.deepSeek.shuwu.Day10.User");Class<User> class2 = User.class;User user = new User("deepSeek", 18);Class<? extends User> class3 = user.getClass();System.out.println(class1);System.out.println(class2);System.out.println(class3);
}

那么这三个class对象是否为同一个呢?

我们来验证一下

java">System.out.println(class1 == class2);
System.out.println(class3 == class2);
System.out.println(class2 == class3);

image-20250312222217020

可以看见这个为ture,哈哈,原来这个是一样的对象

为什么呢?

这时候就要提到大名鼎鼎的JVM

JVM的类加载机制:程序中使用到某个类的时候,需要通过类加载器加载静态文件到JVM中,从而创建对象

每个类只加载一次,JVM每个类只有一个

所以用来描述它的class必然只有一个,所以无论通过那种方式获取class对象,都是同一个对象

Class是整个反射机制的源头

跟反射相关的操作大部分都是基于对象的操作,获取目标类的信息都是通过调用class的相关方法来完成的

类的信息:成员变量、成员方法、构造函数、继承的父类、实现的接口等…

Class的常用方法

方法描述
public native boolean isInterface()判断该类是否为接口
public native boolean isArray()判断该类是否为数组
public boolean isAnnotation()判断该类是否为注解
public String getName()获取该类的全类名
public ClassLoader getClassLoader()获取类加载器
public native Class getSuperclass()获取该类的直接父类
public Package getPackage()获取该类所在的包
public String getPackageName()获取该类所在包的名称
public Class[] getInterfaces()获取该类的全部接口
public native int getModifiers()获取该类的访问权限修饰符
public Field[] getFields()获取该类的全部公有成员变量,包括继承自父类的和自定义的
public Field[] getDeclaredFields()获取该类的自定义成员变量
public Field getField(String name)通过名称获取该类的公有成员变量,包括继承自父类的和自定义的
public Field getDeclaredField(String name)通过名称获取该类的自定义成员变量
public Method[] getMethods()获取该类的全部公有方法,包括继承自父类的和自定义的
public Method[] getDeclaredMethods()获取该类的自定义方法
public Method getMethod(String name,Class… parameterTypes)通过名称和参数信息获取该类的公有方法,包括继承自父类的和自定义的
pyblic Method getDeclaredMethod(String name,Class… parameterTypes)通过名称和参数信息获取该类的自定义方法
public Constructor[] getConstructors()获取该类的公有构造函数
public Constructor[] getDeclaredConstructors()获取该类的全部构造函数
public Constructor getConstructor(Class… parameterTypes)通过参数信息获取该类的公有构造函数
public Consrructor getDeclaredConstructor(Class… parameterTypes)通过参数信息获取该类的构造函数

java中数据类型分为两种:基本类型和引用类型

引用类型一定是对象,而对象一定有Class类

获取类的接口 getInterfaces()

测试User类:

java">public class User implements Serializable{public String name;private int age;public User(String name, int age) {this.name = name;this.age = age;}public String test(String name){return name;}@Overridepublic String toString() {return "User{" +"name='" + name + '\'' +", age=" + age +'}';}
}

main:

java">public static void main(String[] args) throws IOException, ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchFieldException, NoSuchMethodException {Class class1 = Class.forName("com.deepSeek.shuwu.Day10.User");System.out.println(class1.isInterface());Class[] interfaces = class1.getInterfaces();for (Class anInterface : interfaces) {System.out.println(anInterface);}}

输出:
image-20250312230354079

因为获取类可以实现多个接口,所以用数组返回

获取父类 getSuperclass()
public class User extends List {}
java">Class class1 = Class.forName("com.deepSeek.shuwu.Day10.User");
System.out.println(class1.getSuperclass());

image-20250312231350982

获取构造方法

Class 提供了 4 个方法获取构造函数

public Constructor[] getConstructors() :获取所有公有(public)的构造器

public Constructor[] getDeclaredConstructors() :获取所有构造器

public Constructor getConstructor(Class… parameterTypes) :获取单个公有(public)的构造器

public Consrructor getDeclaredConstructor(Class… parameterTypes) :获取单个构造器

反射机制专门提供的一个类 Constructor,专门用来描述类的构造器

Class 专门用来描述类的信息,Constructor 专门用来描述类的构造器

java">public class User {private String name;private int age;public User(String name, int age) {this.name = name;this.age = age;}public User() {}private User(String name){this.name = name;}protected User(int age){this.age = age;}
}
java">public static void main(String[] args) throws IOException, ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchFieldException, NoSuchMethodException {Class class1 = Class.forName("com.deepSeek.shuwu.Day10.User");Constructor[] constructors = class1.getConstructors();for (Constructor constructor : constructors) {System.out.println(constructor);}    public static void main(String[] args) throws IOException, ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchFieldException, NoSuchMethodException {Class class1 = Class.forName("com.deepSeek.shuwu.Day10.User");System.out.println("获取所有公有的构造器");Constructor[] constructors = class1.getConstructors();for (Constructor constructor : constructors) {System.out.println(constructor);}System.out.println("根据参数类型获取单个公有构造器");System.out.println(class1.getConstructor(String.class, int.class));System.out.println("获取所有构造器");for (Constructor constructor : class1.getDeclaredConstructors()) {System.out.println(constructor);}System.out.println("根据参数类型获取单个构造器");System.out.println(class1.getDeclaredConstructor(String.class, int.class));}
}

image-20250312232403026

可以看见公有的有两个,所有的有四个,有Declared则就是所有的


http://www.ppmy.cn/news/1578795.html

相关文章

关于Flutter中两种Widget的生命周期详解

目录 一、StatelessWidget 生命周期 二、StatefulWidget 生命周期 1. 创建阶段 2. State初始化阶段 3. 构建阶段 4. 更新阶段 5. 销毁阶段 三、核心对比与常见陷阱 四、面试回答技巧 以下是Flutter中两种核心Widget&#xff08;StatelessWidget和StatefulWidget&#…

msf(Metasploit)中Session与Channel的区别与关系解析

在 Metasploit Framework&#xff08;MSF&#xff09;中&#xff0c;Session 和 Channel 都是与目标主机的交互方式&#xff0c;但它们的作用和概念有所不同。本文将解析这两个术语的区别。 一、Session&#xff08;会话&#xff09; Session 是指通过 Metasploit 成功利用目标…

状态模式的C++实现示例

核心思想 状态模式&#xff08;State Pattern&#xff09; 是一种行为设计模式&#xff0c;允许对象在其内部状态改变时改变其行为。它将状态相关的逻辑分散到不同的状态类中&#xff0c;避免了使用大量的条件语句来处理不同状态下的行为。 状态抽象化&#xff1a;将对象的状…

单元测试、系统测试和集成测试知识总结

&#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 一、单元测试的概念 单元测试是对软件基本组成单元进行的测试&#xff0c;如函数或一个类的方法。当然这里的基本单元不仅仅指的是一个函数或者方法&#xff…

3.3.2 Proteus第一个仿真图

文章目录 文章介绍0 效果图1 新建“点灯”项目2 添加元器件3 元器件布局接线4 补充 文章介绍 本文介绍&#xff1a;使用Proteus仿真软件画第一个仿真图 0 效果图 1 新建“点灯”项目 修改项目名称和路径&#xff0c;之后一直点“下一步”直到完成 2 添加元器件 点击元…

深入解析Go语言Channel:源码剖析与并发读写机制

文章目录 Channel的内部结构Channel的创建过程有缓冲Channel的并发读写机制同时读写的可能性发送操作的实现接收操作的实现 并发读写的核心机制解析互斥锁保护环形缓冲区等待队列直接传递优化Goroutine调度 实例分析&#xff1a;有缓冲Channel的并发读写性能优化与最佳实践缓冲…

简记_开关电源基础知识(一)

1.1、开关电源的类型 线性稳压器&#xff08;LDO&#xff09; 开关稳压器 电荷泵 传输元件 BJT&#xff08;双极型晶体管&#xff09; FET&#xff08;场效应晶体管&#xff09; BJT或FET 储能元件 电容 电感和电容 电容 工作模式 工作在线性区 无开关跳变 开关&a…

WireShark自动抓包

背景 异常流量检测是当前保护网络空间安全的重要检测方法。 对流量的研究&#xff0c;首先需要在系统中进行抓包&#xff0c;并对包进行分析。 这里对WireShark自动抓包进行简要介绍。 操作步骤 1、选择“捕获”>“选项”。 2、在Input下&#xff0c;选择要抓包的网络接…