java基础之-反射详解

写在前面的话

脑子是个好东西,可惜的是一直没有搞懂脑子的内存删除机制是什么,所以啊,入行多年,零零散散的文章看了无数,却总是学习了很多也忘了很多。

痛定思痛的我决定从今天开始系统的梳理下知识架构,记录下零散的知识,方便温故知新的同时也顺便清除一些大脑空间用来学习更高深的技术。


目录

前言

何为反射?

反射的应用场景

反射机制的优缺点

反射实战

获取 Class 对象的四种方式

反射的一些基本操作

总结


前言

Java的反射(reflection)机制是指在程序的运行状态中,可以构造任意一个类的对象,可以了解任意一个对象所属的类,可以了解任意一个类的成员变量和方法,可以调用任意一个对象的属性和方法。这种动态获取程序信息以及动态调用对象的功能称为Java语言的反射机制。反射被视为动态语言的关键。

何为反射?

反射 (Reflection) 是 Java 的特征之一,它允许运行中的 Java 程序获取自身的信息,并且可以操作类或对象的内部属性。

Oracle 官方对反射的解释是:

Reflection enables Java code to discover information about the fields, methods and constructors of loaded classes, and to use reflected fields, methods, and constructors to operate on their underlying counterparts, within security restrictions.
The API accommodates applications that need access to either the public members of a target object (based on its runtime class) or the members declared by a given class. It also allows programs to suppress default reflective access control.

简而言之,通过反射,我们可以在运行时获得程序或程序集中每一个类型的成员和成员的信息。程序中一般的对象的类型都是在编译期就确定下来的,而 Java 反射机制可以动态地创建对象并调用其属性,这样的对象的类型在编译期是未知的。所以我们可以通过反射机制直接创建对象,即使这个对象的类型在编译期是未知的。

反射的核心是 JVM 在运行时才动态加载类或调用方法/访问属性,它不需要事先(写代码的时候或编译期)知道运行对象是谁。

Java 反射主要提供以下功能:

  • 在运行时判断任意一个对象所属的类;
  • 在运行时构造任意一个类的对象;
  • 在运行时判断任意一个类所具有的成员变量和方法(通过反射甚至可以调用private方法);
  • 在运行时调用任意一个对象的方法

🎗️重点:是运行时而不是编译时

反射的应用场景

像咱们平时大部分时候都是在写业务代码,很少会接触到直接使用反射机制的场景。

但是,这并不代表反射没有用。相反,正是因为反射,你才能这么轻松地使用各种框架。像 Spring/Spring Boot、MyBatis 等等框架中都大量使用了反射机制。

这些框架中也大量使用了动态代理,而动态代理的实现也依赖反射。

比如下面是通过 JDK 实现动态代理的示例代码,其中就使用了反射类 Method 来调用指定的方法。


  
  
  1. public class DebugInvocationHandler implements InvocationHandler {
  2. /**
  3. * 代理类中的真实对象
  4. */
  5. private final Object target;
  6. public DebugInvocationHandler (Object target) {
  7. this.target = target;
  8. }
  9. public Object invoke (Object proxy, Method method, Object[] args) throws InvocationTargetException, IllegalAccessException {
  10. System.out.println( "before method " + method.getName());
  11. Object result = method.invoke(target, args);
  12. System.out.println( "after method " + method.getName());
  13. return result;
  14. }
  15. }

另外,像 Java 中的一大利器 注解 的实现也用到了反射。

为什么你使用 Spring 的时候 ,一个@Component注解就声明了一个类为 Spring Bean 呢?为什么你通过一个 @Value注解就读取到配置文件中的值呢?究竟是怎么起作用的呢?

这些都是因为你可以基于反射分析类,然后获取到类/属性/方法/方法的参数上的注解。你获取到注解之后,就可以做进一步的处理。

反射机制的优缺点

优点 : 可以让咱们的代码更加灵活、为各种框架提供开箱即用的功能提供了便利

缺点 :让我们在运行时有了分析操作类的能力,这同样也增加了安全问题。比如可以无视泛型参数的安全检查(泛型参数的安全检查发生在编译时)。另外,反射的性能也要稍差点,不过,对于框架来说实际是影响不大的。相关阅读:Java Reflection: Why is it so slow?open in new window

反射实战

获取 Class 对象的四种方式

如果我们动态获取到这些信息,我们需要依靠 Class 对象。Class 类对象将一个类的方法、变量等信息告诉运行的程序。Java 提供了四种方式获取 Class 对象:

1. 知道具体类的情况下可以使用:

Class alunbarClass = TargetObject.class;

  
  

但是我们一般是不知道具体类的,基本都是通过遍历包下面的类来获取 Class 对象,通过此方式获取 Class 对象不会进行初始化

2. 通过 Class.forName()传入类的全路径获取:

Class alunbarClass1 = Class.forName("cn.javaguide.TargetObject");

  
  

3. 通过对象实例instance.getClass()获取:


  
  
  1. TargetObject o = new TargetObject();
  2. Class alunbarClass2 = o.getClass();

4. 通过类加载器xxxClassLoader.loadClass()传入类路径获取:

ClassLoader.getSystemClassLoader().loadClass("cn.javaguide.TargetObject");

  
  

通过类加载器获取 Class 对象不会进行初始化,意味着不进行包括初始化等一系列步骤,静态代码块和静态对象不会得到执行

反射的一些基本操作

  1. 创建一个我们要使用反射操作的类 TargetObject

  
  
  1. package cn.javaguide;
  2. public class TargetObject {
  3. private String value;
  4. public TargetObject () {
  5. value = "JavaGuide";
  6. }
  7. public void publicMethod (String s) {
  8. System.out.println( "I love " + s);
  9. }
  10. private void privateMethod () {
  11. System.out.println( "value is " + value);
  12. }
  13. }
  1. 使用反射操作这个类的方法以及参数

  
  
  1. package cn.javaguide;
  2. import java.lang.reflect.Field;
  3. import java.lang.reflect.InvocationTargetException;
  4. import java.lang.reflect.Method;
  5. public class Main {
  6. public static void main (String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException, NoSuchFieldException {
  7. /**
  8. * 获取 TargetObject 类的 Class 对象并且创建 TargetObject 类实例
  9. */
  10. Class<?> targetClass = Class.forName( "cn.javaguide.TargetObject");
  11. TargetObject targetObject = (TargetObject) targetClass.newInstance();
  12. /**
  13. * 获取 TargetObject 类中定义的所有方法
  14. */
  15. Method[] methods = targetClass.getDeclaredMethods();
  16. for (Method method : methods) {
  17. System.out.println(method.getName());
  18. }
  19. /**
  20. * 获取指定方法并调用
  21. */
  22. Method publicMethod = targetClass.getDeclaredMethod( "publicMethod",
  23. String.class);
  24. publicMethod.invoke(targetObject, "JavaGuide");
  25. /**
  26. * 获取指定参数并对参数进行修改
  27. */
  28. Field field = targetClass.getDeclaredField( "value");
  29. //为了对类中的参数进行修改我们取消安全检查
  30. field.setAccessible( true);
  31. field.set(targetObject, "JavaGuide");
  32. /**
  33. * 调用 private 方法
  34. */
  35. Method privateMethod = targetClass.getDeclaredMethod( "privateMethod");
  36. //为了调用private方法我们取消安全检查
  37. privateMethod.setAccessible( true);
  38. privateMethod.invoke(targetObject);
  39. }
  40. }

输出内容:


  
  
  1. public Method
  2. privateMethod
  3. I love JavaGuide
  4. value is JavaGuide

注意 : 有读者提到上面代码运行会抛出 ClassNotFoundException 异常,具体原因是你没有下面把这段代码的包名替换成自己创建的 TargetObject 所在的包 。

Class<?> targetClass = Class.forName("cn.javaguide.TargetObject");
  
  

总结

反射是java很重要的特性之一,了解反射可以让我们更好的进行编程

转自:反射机制详解 | JavaGuide

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值