Java中的Reflect

本文详细介绍Java反射机制的核心概念,包括如何获取Class实例、利用Class实例获取类信息、实例化对象及调用方法等。此外还介绍了如何使用反射进行动态代理,并提供了一些高级用法如绕过访问控制。

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

Java 中反射总的来说就是获得Class实例,通过这个Class实例来获取类的信息从而可以实例化该类或者调用方法

1. 获得Class实例

一共有三种方法获得


Class clazz = Class.forName("cn.sheep3.efjava.reflect.Persen");

Class clazz = Demo.class;

Class clazz = demo.getClass();
2. 通过Class实例获得类信息

//获得构造器

clazz.getConstructors();

//获得接口

clazz.getInterfaces();

//获得注解

clazz.getAnnotations()

//获得父类

clazz.getSuperclass()

//获得修饰符

clazz.getModifiers()

//获得方法

clazz.getMethods()
3. getDeclaredXxx 与 getXxx
  • getDeclaredXxx 获得自身所有的Xxx

  • getXxx 获得所有Xxx包括父类

4. 实例化对象

//两种方式

//java.lang.Class.newInstance()

Object obj = Class.forName("java.lang.Object").newInstance();

//java.lang.reflect.Constructor.newInstance()

Constructor<?>[] cons = clazz.getConstructors();
per1 = (Person) cons[0].newInstance();
5. 调用方法

//先获得Method实例

Method declaredMethod = clazz.getDeclaredMethod("getAge");

//调用Method实例的invoke方法,第一个参数是对象实例,后面的可变参数是方法参数

Object m_return = declaredMethod.invoke(person);
6. 绕过访问控制

只需要在获取到Constructor、Field和Method类的对象之后,调用setAccessible方法并设为true即可

7. 动态代理
  1. 实现InvocationHandler接口

  2. 重写invoke(),在这里hook目标方法

  3. 获得proxy实例(List) Proxy.newProxyInstance(类加载器, Class[], InvocationHandler的实现)

  4. 通过这个proxy来调用方法

### Java 反射 Reflect 使用教程 #### 获取 `Class` 对象 在Java中,反射的核心是`Class`对象。可以通过三种方式获得: - **通过类名**:使用`.class`语法。 ```java Class<?> clazz = String.class; ``` - **通过实例**:调用对象的`.getClass()`方法。 ```java Object obj = new Integer(0); Class<?> clazz = obj.getClass(); ``` - **通过全限定名**:利用`Class.forName(String className)`静态方法。 ```java try { Class<?> clazz = Class.forName("java.lang.String"); } catch (ClassNotFoundException e) { // Handle exception } ``` 上述每种方法都可用于获取指定类型的`Class`对象[^1]。 #### 访问字段 (`Field`) 一旦有了`Class`对象,就可以访问其声明的成员变量(即字段)。这包括私有字段在内的所有字段。 ```java // 假设有一个Person类 public class Person { private String name; public void setName(String n){ this.name=n; } @Override public String toString(){ return "Name:"+this.name; } } try { Class<Person> personClass = Person.class; // 获取名为"name"的私有字段 Field field = personClass.getDeclaredField("name"); // 设置可访问标志位为true以便能够读写该字段 field.setAccessible(true); Person p = new Person(); // 向p设置值 field.set(p,"John Doe"); System.out.println(p.toString()); // 输出 Name:John Doe } catch (NoSuchFieldException | IllegalAccessException e) { // 处理异常... } ``` 这段代码展示了如何创建一个新的人(`Person`)对象并为其内部不可见的名字(`name`)属性赋值。 #### 调用方法 (`Method`) 除了可以操作字段外,还可以通过反射来执行某个特定的方法。 ```java try { Class<Person> personClass = Person.class; Method method = personClass.getMethod("setName",String.class); Person p = new Person(); // 执行setName()方法并将参数传递给它 method.invoke(p, "Jane Doe"); System.out.println(p.toString()); } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) { // 处理异常... } ``` 这里说明了怎样找到公共方法`setName`并通过传入的对象实例去调用这个方法。 #### 构造器 (`Constructor`) 最后,也可以借助于反射机制来进行对象的构建工作。 ```java try { Class<Person> personClass = Person.class; Constructor<Person> constructor = personClass.getConstructor(); Person p = constructor.newInstance(); System.out.println(p.toString()); } catch (InstantiationException | IllegalAccessException | InvocationTargetException | NoSuchMethodException e) { // 处理异常... } ``` 此片段显示了一个无参构造函数是如何被用来生成新的`Person`实体的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值