反射Reflection

1. 反射的出现背景

        Java 程序中,所有的对象都有两种类型:编译时类型运行时类型,而很多时候对象的编译时类型和运行时类型不一致。 

        例如:某些变量或形参的声明类型是 Object 类型,但是程序却需要调用该对象运行时类型的方法,该方法不是 Object 中的方法,那么如何解决呢?

        解决这个问题,有两种方案:

        方案 1:在编译和运行时都完全知道类型的具体信息,在这种情况下,我们可以直接先使用 instanceof 运算符进行判断,再利用强制类型转换符将其转换成运行时类型的变量即可。

        方案 2:编译时根本无法预知该对象和类的真实信息,程序只能依靠运行时信息来发现该对象和类的真实信息,这就必须使用反射。


2. 反射概述

        Reflection(反射)是被视为动态语言的关键,反射机制允许程序在运行期间借助于 Reflection API 取得任何类的内部信息,并能直接操作任意对象的内部属性及方法。

        加载完类之后,在堆内存的方法区中就产生了一个 Class 类型的对象(一个类只有一个 Class 对象),这个对象就包含了完整的类的结构信息。我们可以通过这个对象看到类的结构。这个对象就像一面镜子,透过这个镜子看到类的结构, 所以,我们形象的称之为:反射。

         从内存加载上看反射:

public class Test1 {public static void main(String[] args) throws InstantiationException, IllegalAccessException, NoSuchFieldException, SecurityException, NoSuchMethodException, InvocationTargetException {// 反射// 创建实例Class<Person> clazz = Person.class;Person person2 = clazz.newInstance();System.out.println(person2);// 调用属性Field ageField = clazz.getField("age");ageField.set(person2, 10);System.out.println(ageField.get(person2));// 调用方法Method showMethod = clazz.getMethod("show");showMethod.invoke(person2);// 调用私有构造器Class<Person> clazz1 = Person.class;Constructor<Person> cons = clazz1.getDeclaredConstructor(String.class, int.class);cons.setAccessible(true);Person person3 = (Person) cons.newInstance("Tom", 22);System.out.println(person3);// 调用私有属性Field namField = clazz1.getDeclaredField("name");namField.setAccessible(true);namField.set(person3, "Jerry");System.out.println(person3);}}class Person {private String name;public int age;public static boolean sex = false;public Person() {System.out.println("Person()...");}@SuppressWarnings("unused")private Person(String name, int age) {System.out.println("Person()..." + name + age);this.name = name;this.age = age;}public void show() {System.out.println("Person.show()...");}private String m1(String name, int age) {System.out.println("m1()..." + name + "-" + age);return "m1()..." + name + "-" + age;}@Overridepublic String toString() {return "Person [name=" + name + ", age=" + age + "]";} 
}

3. Java 反射机制功能及优缺点

Java 反射机制提供的功能:

  • 在运行时判断任意一个对象所属的类
  • 在运行时构造任意一个类的对象
  • 在运行时判断任意一个类所具有的成员变量和方法
  • 在运行时获取泛型信息
  • 在运行时调用任意一个对象的成员变量和方法
  • 在运行时处理注解
  • 生成动态代理

优点:

  • 提高了 Java 程序的灵活性和扩展性,降低了耦合性,提高自适应能力
  • 允许程序创建和控制任何类的对象,无需提前硬编码目标类

缺点:

  • 反射的性能较低,反射机制主要应用在对灵活性和扩展性要求很高的系统框架上
  • 反射会模糊程序内部逻辑,可读性较差

4. 反射的应用

4.1 创建运行时类的对象

方式 1:直接调用 Class 对象的 newInstance() 方法

步骤:

  1. 获取该类型的 Class 对象
  2. 调用 Class 对象的 newInstance()方法创建对象

要求:

  • 类必须有一个无参数的构造器
  • 类的构造器的访问权限需要足够
Class<Person> clazz = Person.class;
Person p = clazz.newInstance();

方式 2:通过获取构造器对象来进行实例化

步骤:

  1. 通过 Class 类的 getDeclaredConstructor(Class … parameterTypes)取得本类的指定形参类型的构造器
  2. 向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数。
  3. 通过 Constructor 实例化对象

如果构造器的权限修饰符修饰的范围不可见,可以调用 setAccessible(true)   

Class<Person> clazz1 = Person.class;
Constructor<Person> cons = clazz1.getDeclaredConstructor(String.class, int.class);
cons.setAccessible(true);
Person p1 = (Person) cons.newInstance("Tom", 22);

4.2 获取运行时类的完整结构

可以获取:包、修饰符、类型名、父类(包括泛型父类)、父接口(包括泛型父接口)、成员(属性、构造器、方法)、注解(类上的、方法上的、属性上的)。

相关API:

//1.实现的全部接口 
public Class<?>[] getInterfaces() 
//确定此对象所表示的类或接口实现的接口。 //2.所继承的父类 
public Class<? Super T> getSuperclass() 
//返回表示此 Class 所表示的实体(类、接口、基本类型)的父类的 Class。 //3.全部的构造器 
public Constructor<T>[] getConstructors() 
//返回此 Class 对象所表示的类的所有 public 构造方法。 
public Constructor<T>[] getDeclaredConstructors() 
//返回此 Class 对象表示的类声明的所有构造方法。 //Constructor 类中: 
//取得修饰符: 
public int getModifiers(); 
//取得方法名称: 
public String getName(); 
//取得参数的类型: 
public Class<?>[] getParameterTypes(); //4.全部的方法 
public Method[] getDeclaredMethods() 
//返回此 Class 对象所表示的类或接口的全部方法 
public Method[] getMethods() 
//返回此 Class 对象所表示的类或接口的 public 的方法 //Method 类中: 
public Class<?> getReturnType() 
//取得全部的返回值 
public Class<?>[] getParameterTypes() 
//取得全部的参数 
public int getModifiers() 
//取得修饰符 
public Class<?>[] getExceptionTypes() 
//取得异常信息 //5.全部的 Field 
public Field[] getFields() 
//返回此 Class 对象所表示的类或接口的 public 的 Field。 
public Field[] getDeclaredFields() 
//返回此 Class 对象所表示的类或接口的全部 Field。 //Field 方法中: 
public int getModifiers() 
//以整数形式返回此 Field 的修饰符 
public Class<?> getType() 
//得到 Field 的属性类型 
public String getName() 
//返回 Field 的名称。 //6. Annotation 相关 
get Annotation(Class<T> annotationClass) 
getDeclaredAnnotations() //7.泛型相关 
//获取父类泛型类型: 
Type getGenericSuperclass() 
//泛型类型:ParameterizedType 
//获取实际的泛型类型参数数组: 
getActualTypeArguments() //8.类所在的包 
Package getPackage()

4.3 调用运行时类的指定结构

调用指定构造器
Class<Person> clazz1 = Person.class;
Constructor<Person> cons = clazz1.getDeclaredConstructor(String.class, int.class);
cons.setAccessible(true);
Person p1 = (Person) cons.newInstance("Tom", 22);
调用指定属性

(1)获取该类型的 Class 对象 Class clazz = Class.forName("包.类名");

(2)获取属性对象 Field field = clazz.getDeclaredField("属性名");

(3)如果属性的权限修饰符不是 public,那么需要设置属性可访问 field.setAccessible(true);

(4)创建实例对象:如果操作的是非静态属性,需要创建实例对象 Object obj = clazz.newInstance();有公共的无参构造 Object obj = 构造器对象.newInstance(实参...);通过特定构造器对象创建实例对象。

(5)设置指定对象 obj 上此 Field 的属性内容 field.set(obj,"属性值");如果操作静态变量,那么实例对象可以省略,用 null 该类型的 Class 对象表示。

(6)取得指定对象 obj 上此 Field 的属性内容 Object value = field.get(obj);如果操作静态变量,那么实例对象可以省略,用 null 或该类型的 Class 对象表示。

// public 属性
Field ageField = clazz1.getField("age");
ageField.set(p1, 3);
System.out.println(ageField.get(p1));   // 3// private 属性
Field namField = clazz1.getDeclaredField("name");
namField.setAccessible(true);
namField.set(p1, "xiaoLI");
System.out.println(namField.get(p1));   // xiaoLI// static 属性
Field sexField = clazz1.getDeclaredField("sex");
sexField.set(clazz1, true);
System.out.println(sexField.get(clazz1));   // true
调用指定方法

(1)获取该类型的 Class 对象 Class clazz = Class.forName("包.类名");

(2)获取方法对象 Method method = clazz.getDeclaredMethod("方法名",方法的形参类型列表);

(3)创建实例对象 Object obj = clazz.newInstance();

(4)调用方法 Object result = method.invoke(obj, 方法的实参值列表);如果方法的权限修饰符修饰的范围不可见,也可以调用 setAccessible(true) 。如果方法是静态方法,实例对象也可以省略,用 null 代替。

// 方法
Method m1Method = clazz1.getDeclaredMethod("m1", String.class, int.class);
m1Method.setAccessible(true);
Object returnVal = m1Method.invoke(p1, "YY", 18);
System.out.println((String)returnVal);  // m1()...YY-18
读取注解信息

一个完整的注解应该包含三个部分: (1)声明 (2)使用 (3)读取

@Inherited 
@Target(ElementType.TYPE) 
@Retention(RetentionPolicy.RUNTIME) 
public @interface Anno {String value(); 
}@Anno(value = "t_person")
class Person {...
}// 获取类上的注解
Anno an = clazz1.getDeclaredAnnotation(Anno.class);
System.out.println(an.value());     // t_person

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/589526.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java编程使用CGLIB动态代理介绍与实战演示

文章目录 前言技术积累核心概念主要功能适用场景与JDK动态代理的对比 实战演示定义待代理的目标类实现MethodInterceptor接口使用代理对象 测试结果写在最后 前言 在Java编程中&#xff0c;CGLIB (Code Generation Library) 是一个强大的高性能代码生成库&#xff0c;它通过生…

WebGIS 地铁交通线网 | 图扑数字孪生

数字孪生技术在地铁线网的管理和运维中的应用是一个前沿且迅速发展的领域。随着物联网、大数据、云计算以及人工智能技术的发展&#xff0c;地铁线网数字孪生在智能交通和智慧城市建设中的作用日益凸显。 图扑软件基于 HTML5 的 2D、3D 图形渲染引擎&#xff0c;结合 GIS 地图…

[RK3588-Android12] 调试MIPI-双通道-压缩屏(Video Mode/MIPI Dphy 8Lane/DSC 144HZ)

问题描述 被测屏幕&#xff1a;小米Pad6 分辨率&#xff1a;1800X2880 模式&#xff1a;Video Mode/MIPI Dphy 8Lane/DSC 144HZ PPS: 11 00 00 89 30 80 0B 40 03 84 00 14 01 C2 01 C2 02 00 01 F4 00 20 01 AB 00 06 00 0D 05 7A 06 1A 18 00 10 F0 03 0C 20 00 06 0B 0B 33…

【c++】STl-list使用list模拟实现

主页&#xff1a;醋溜马桶圈-CSDN博客 专栏&#xff1a;c_醋溜马桶圈的博客-CSDN博客 gitee&#xff1a;mnxcc (mnxcc) - Gitee.com 目录 1. list的介绍及使用 1.1 list的介绍 1.2 list的使用 1.2.1 list的构造 1.2.2 list iterator的使用 1.2.3 list capacity 1.2.4 …

【智能算法】蜜獾算法(HBA)原理及实现

目录 1.背景2.算法原理2.1算法思想2.2算法过程 3.结果展示4.参考文献 1.背景 2021年&#xff0c;FA Hashim等人受到自然界中蜜獾狩猎行为启发&#xff0c;提出了蜜獾算法&#xff08;(Honey Badger Algorithm&#xff0c;HBA&#xff09;。 2.算法原理 2.1算法思想 蜜獾以其…

pytest--python的一种测试框架--接口测试

接口测试 工具&#xff1a; POSTMAN&#xff1b; 接口选择&#xff1a; 豆瓣电影&#xff0c;进制数据 POSTMAN下载&#xff1a; 1.POSTMAN官网&#xff1a;https://www.postman.com/products/&#xff1b; 2.点product选Download Postman 下载完之后双击打开就可以用的。…

软件设计师27--规范化理论

软件设计师27--规范化理论 考点1&#xff1a;规范化理论基本概念函数依赖规范化理论--Amstrong公理体系候选键主属性与非主属性例题&#xff1a; 考点2&#xff1a;范式判断规范化理论规范化理论 - 范式例题&#xff1a; 考点3&#xff1a;范式分解保持函数依赖分解无损分解模式…

简单了解波 Mono-repo Multi-repo(Poly-repo)

Mono-repo 和 Multi-repo 是软件开发中代码管理的两个不同策略。Mono-repo & Multi-repo 孰优孰劣是个老生常谈得话题了&#xff0c;这里就不 PK 了&#xff0c;“略微”看下两者区别。 当我们使用 Git 作为版本控制系统管理项目的代码时&#xff0c;那么 monorepo 与 mul…

实现顺序表(增、删、查、改)

引言&#xff1a;顺序表是数据结构中的一种形式&#xff0c;就是存储数据的一种结构。 这里会用到动态内存开辟&#xff0c;指针和结构体的知识 1.什么是数据结构 数据结构就是组织和存储数据的结构。 数据结构的特性&#xff1a; 物理结构&#xff1a;在内存中存储的数据是否连…

Linux系统中的动态追踪技术

在Linux系统中&#xff0c;动态追踪技术是一种强大的工具&#xff0c;用于在运行时追踪和分析系统和应用程序的行为。动态追踪技术允许用户监视系统和应用程序的运行&#xff0c;捕获关键信息&#xff0c;以便于调试、性能优化、故障排除和安全监控。本文将介绍Linux系统中的动…

linux设置Nacos自启动

前提&#xff1a;已经安装好nacos应用 可参考&#xff1a;Nacos单机版安装-CSDN博客 1. 创建nacos.service 1.1 在 /lib/systemd/system 目录底下&#xff0c;新建nacos.service文件 [Unit] Descriptionnacos Afternetwork.target[Service]Typeforking# 单机启动方式&#…

肿瘤免疫反应瀑布图(源于The Miller Lab)

目录 数据格式 绘图 ①根据剂量 ②根据type ③根据治疗响应度 添加水平线 数据格式 肿瘤免疫响应数据 rm(list ls()) library(tidyverse) library(dplyr) library(knitr)#模拟数据 # We will randomly assign the two doses, 80 mg or 150 mg, to the 56 subjects Me…