当前位置: 首页 > news >正文

Java中的反射是怎么回事?

反射的概念

《Java核心技术》中的定义是这样的:能够分析类能力的程序,就是反射

这就是一个概念,跟java中经常提的问题:对象是什么? 一类问题,简单来说就是将类创建对象的逻辑反过来,由对象获得其所属类型的信息。它允许程序在运行时查询和操作类、方法、字段等信息。

实现原理机制

说到反射就不得不谈一个Class类,就是类的类,存的是各种类的状态信息的类,Java的反射机制主要通过java.lang.Class类来实现。你可以通过Class.forName("类名")方法来获取一个Class对象,进而获取该类的信息。

  • Object 类中的getClass( ) 方法将会返回一个Class 类型的实例。
  • Class类常用的方法。
    • e.getname()获取e的类的名字
    • 静态方法Class.forName(className),返回对应的Class对象。
    • 来动态地创建一个类的实例e.getClass().newInstance();

 利用反射获取打印类的信息(也叫能力)

不是通过“ . ”的形式,获得一个类的方法,属性,等具体信息

package test729;import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.Scanner;public class ReflectionTest {/*** 判断程序启动时是否有命令行参数。如果有,则将第一个参数作为类名;如果没有,则提示用户输入类名并读取用户输入。* @param args*/public static void main(String[] args) {String className;if (args.length > 0) {className = args[0];} else {Scanner scanner = new Scanner(System.in);System.out.println("请输入类名(如:java.util.Date):");className = scanner.next();}
/***使用 Class.forName(className) 动态加载指定的类。然后获取该类的父类和修饰符。* 打印类的修饰符、类名和父类信息。* 调用 printConstructors(cl), printMethods(cl), 和 printFields(cl) 方法,分别打印构造函数、方法和字段的信息。* 捕捉并处理 ClassNotFoundException 异常,打印异常信息。*/try {Class<?> cl = Class.forName(className);Class<?> superClass = cl.getSuperclass();String modifiers = Modifier.toString(cl.getModifiers());if (modifiers.length() > 0) {System.out.print(modifiers + " ");}System.out.print("class " + className);if (superClass != null && superClass != Object.class) {System.out.print(" extends " + superClass.getName());}System.out.println("\n{");printConstructors(cl);System.out.println();printMethods(cl);System.out.println();printFields(cl);System.out.println("}");} catch (ClassNotFoundException e) {e.printStackTrace();}}/*** printConstructors 方法打印类的构造函数信息。* 使用 cl.getDeclaredConstructors() 获取所有构造函数。* 遍历每个构造函数,获取其修饰符、名称和参数类型。* 打印构造函数的修饰符、名称和参数列表。* @param cl*/public static void printConstructors(Class<?> cl) {Constructor<?>[] constructors = cl.getDeclaredConstructors();for (Constructor<?> c : constructors) {String name = c.getName();String modifiers = Modifier.toString(c.getModifiers());System.out.print("  ");if (modifiers.length() > 0) {System.out.print(modifiers + " ");}System.out.print(name + "(");Class<?>[] paramTypes = c.getParameterTypes();for (int i = 0; i < paramTypes.length; i++) {if (i > 0) {System.out.print(", ");}System.out.print(paramTypes[i].getName());}System.out.println(");");}}/*** printMethods 方法打印类的方法信息。* 使用 cl.getDeclaredMethods() 获取所有方法。* 遍历每个方法,获取其修饰符、返回类型、名称和参数类型。* 打印方法的修饰符、返回类型、名称和参数列表。* @param cl*/public static void printMethods(Class<?> cl) {Method[] methods = cl.getDeclaredMethods();for (Method method : methods) {Class<?> returnType = method.getReturnType();String name = method.getName();String modifiers = Modifier.toString(method.getModifiers());System.out.print("  ");if (modifiers.length() > 0) {System.out.print(modifiers + " ");}System.out.print(returnType.getName() + " " + name + "(");Class<?>[] paramTypes = method.getParameterTypes();for (int i = 0; i < paramTypes.length; i++) {if (i > 0) {System.out.print(", ");}System.out.print(paramTypes[i].getName());}System.out.println(");");}}/*** printFields 方法打印类的字段信息。* 使用 cl.getDeclaredFields() 获取所有字段。* 遍历每个字段,获取其修饰符、类型和名称。* 打印字段的修饰符、类型和名称。* 以上代码整体上实现了动态加载类,并输出类的构造函数、方法和字段的信息。* @param cl*/public static void printFields(Class<?> cl) {Field[] fields = cl.getDeclaredFields();for (Field f : fields) {Class<?> type = f.getType();String name = f.getName();String modifiers = Modifier.toString(f.getModifiers());System.out.print("  ");if (modifiers.length() > 0) {System.out.print(modifiers + " ");}System.out.println(type.getName() + " " + name + ";");}}
}

 这个测试案例可以利用反射打印出一个类的所有信息和属性。

测试输出结果

请输入类名(如:java.util.Date):
java.util.Date
public class java.util.Date
{public java.util.Date(int, int, int, int, int, int);public java.util.Date(java.lang.String);public java.util.Date();public java.util.Date(long);public java.util.Date(int, int, int);public java.util.Date(int, int, int, int, int);public boolean after(java.util.Date);public boolean before(java.util.Date);public boolean equals(java.lang.Object);public java.lang.String toString();public int hashCode();public java.lang.Object clone();public int compareTo(java.util.Date);public volatile int compareTo(java.lang.Object);private void readObject(java.io.ObjectInputStream);private void writeObject(java.io.ObjectOutputStream);private final sun.util.calendar.BaseCalendar$Date normalize(sun.util.calendar.BaseCalendar$Date);private final sun.util.calendar.BaseCalendar$Date normalize();public static long parse(java.lang.String);public static java.util.Date from(java.time.Instant);public long getTime();public void setTime(long);public int getDate();public static long UTC(int, int, int, int, int, int);private static final java.lang.StringBuilder convertToAbbr(java.lang.StringBuilder, java.lang.String);private final sun.util.calendar.BaseCalendar$Date getCalendarDate();private static final sun.util.calendar.BaseCalendar getCalendarSystem(int);private static final sun.util.calendar.BaseCalendar getCalendarSystem(sun.util.calendar.BaseCalendar$Date);private static final sun.util.calendar.BaseCalendar getCalendarSystem(long);public int getDay();public int getHours();private static final synchronized sun.util.calendar.BaseCalendar getJulianCalendar();static final long getMillisOf(java.util.Date);public int getMinutes();public int getMonth();public int getSeconds();private final long getTimeImpl();public int getTimezoneOffset();public int getYear();public void setDate(int);public void setHours(int);public void setMinutes(int);public void setMonth(int);public void setSeconds(int);public void setYear(int);public java.lang.String toGMTString();public java.time.Instant toInstant();public java.lang.String toLocaleString();private static final sun.util.calendar.BaseCalendar gcal;private static sun.util.calendar.BaseCalendar jcal;private transient long fastTime;private transient sun.util.calendar.BaseCalendar$Date cdate;private static int defaultCenturyStart;private static final long serialVersionUID;private static final [Ljava.lang.String; wtb;private static final [I ttb;
}进程已结束,退出代码为 0
http://www.lryc.cn/news/409622.html

相关文章:

  • 07 STM32寄存器开发基础-中断编程
  • 聚簇和非聚簇索引/Btree和B+tree
  • 清华学姐熬夜肝了15天的软件测试面试题出炉(附答案)建议收藏!
  • Docker 安装指南
  • 系统架构设计师 - 知识产权与标准化
  • 【Python】Facebook开源时间序列数据预测模型Prophet
  • Spring 常用的三种拦截器详解
  • 微前端概念
  • FFmpeg实战 - 解复用解码
  • Jmeter混合压测(2407)
  • Prometheus各类监控及监控指标和告警规则
  • G120 EPos配置方案及应用场景
  • 定制化爬虫管理:为企业量身打造的数据抓取方案
  • Javascript面试基础6【每日更新10】
  • CTF Web信息搜集 25000字详解
  • MSPM0G3507之电赛小车
  • linux运维一天一个shell命令之vmstat详解
  • 前端开发调试工具推荐分类整理
  • http协议与nginx
  • 一款国外开发的高质量WordPress下载站模板主题
  • Laravel为什么会成为最优雅的PHP框架
  • 孤儿进程的例子
  • CSS前端面试题——怎么用CSS实现一个宽高自适应的正方形?
  • 谷粒商城实战笔记-56~57-商品服务-API-三级分类-修改-拖拽功能完成
  • Shader入门精要总结(二)矩阵
  • 基于CentOS Stream 9平台安装MySQL Community Server 9.0.1 Innovation
  • 正则采集器之五——商品匹配规则
  • 一键切换阿里yum源(包括其他系统repo镜像查找方法)
  • Fiddler学习笔记
  • 【Vue3】watch 监视多种类型数据