深入解析java反射(下)

本文涉及的产品
云解析 DNS,旗舰版 1个月
云解析DNS,个人版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 本博文主要记录Java 反射(reflect)的使用,在了解反射之前,你应该先了解 Java 中的 Class 类,如果你不是很了解,可以先简单了解下。

3.6、获取类的所有方法

修改实体类

/**
 * 基类
 */
public class BaseClass {
  public String publicBaseVar1;
  public String publicBaseVar2;
  private void privatePrintBaseMsg(String var) {
    System.out.println("基类-私有方法,变量:" + var);
  }
  public void publicPrintBaseMsg(String var) {
    System.out.println("基类-公共方法,变量:" + var);
  }
}
/**
 * 子类
 */
public class ChildClass extends BaseClass{
  public String publicOneVar1;
  public String publicOneVar2;
  private String privateOneVar1;
  private String privateOneVar2;
  public String printOneMsg() {
    return privateOneVar1;
  }
  private void privatePrintOneMsg(String var) {
    System.out.println("子类-私有方法,变量:" + var);
  }
  public void publicPrintOneMsg(String var) {
    System.out.println("子类-公共方法,变量:" + var);
  }
}

测试:

public class MethodTest {
  public static void main(String[] args) {
    //1.获取并输出类的名称
    Class mClass = ChildClass.class;
    System.out.println("类的名称:" + mClass.getName());
    System.out.println("----获取所有 public 访问权限的方法,包括自己声明和从父类继承的---");
    //2 获取所有 public 访问权限的方法,包括自己声明和从父类继承的
    Method[] mMethods = mClass.getMethods();
    for (Method method : mMethods) {
      //获取并输出方法的访问权限(Modifiers:修饰符)
      int modifiers = method.getModifiers();
          System.out.print(Modifier.toString(modifiers) + " ");
          //获取并输出方法的返回值类型
          Class returnType = method.getReturnType();
          System.out.print(returnType.getName() + " " + method.getName() + "( ");
          //获取并输出方法的所有参数
          Parameter[] parameters = method.getParameters();
          for (Parameter parameter : parameters) {
              System.out.print(parameter.getType().getName() + " " + parameter.getName() + ",");
          }
          //获取并输出方法抛出的异常
          Class[] exceptionTypes = method.getExceptionTypes();
          if (exceptionTypes.length == 0){
              System.out.println(" )");
          } else {
              for (Class c : exceptionTypes) {
                  System.out.println(" ) throws " + c.getName());
              }
          }
    }
    System.out.println("----获取所有本类的的方法---");
    //3. 获取所有本类的的方法
      Method[] allMethods = mClass.getDeclaredMethods();
      for (Method method : allMethods) {
      //获取并输出方法的访问权限(Modifiers:修饰符)
      int modifiers = method.getModifiers();
      System.out.print(Modifier.toString(modifiers) + " ");
          //获取并输出方法的返回值类型
          Class returnType = method.getReturnType();
          System.out.print(returnType.getName() + " " + method.getName() + "( ");
          //获取并输出方法的所有参数
          Parameter[] parameters = method.getParameters();
          for (Parameter parameter : parameters) {
              System.out.print(parameter.getType().getName() + " " + parameter.getName() + ",");
          }
          //获取并输出方法抛出的异常
          Class[] exceptionTypes = method.getExceptionTypes();
          if (exceptionTypes.length == 0){
              System.out.println(" )");
          } else {
              for (Class c : exceptionTypes) {
                  System.out.println(" ) throws " + c.getName());
              }
          }
    }
  }
}

输出:

类的名称:com.example.java.reflect.ChildClass
----获取所有 public 访问权限的方法,包括自己声明和从父类继承的---
public java.lang.String printOneMsg(  )
public void publicPrintOneMsg( java.lang.String arg0, )
public void publicPrintBaseMsg( java.lang.String arg0, )
public final void wait( long arg0,int arg1, ) throws java.lang.InterruptedException
public final native void wait( long arg0, ) throws java.lang.InterruptedException
public final void wait(  ) throws java.lang.InterruptedException
public boolean equals( java.lang.Object arg0, )
public java.lang.String toString(  )
public native int hashCode(  )
public final native java.lang.Class getClass(  )
public final native void notify(  )
public final native void notifyAll(  )
----获取所有本类的的方法---
public java.lang.String printOneMsg(  )
private void privatePrintOneMsg( java.lang.String arg0, )
public void publicPrintOneMsg( java.lang.String arg0, )

为啥会输出这么多呢?

因为所有的类默认继承object类,打开object类会发现有些公共的方法,所以一并打印出来了!

3.7、调用方法

public class MethodInvokeTest {
  public static void main(String[] args) throws Exception {
    // 1.获取并输出类的名称
    Class mClass = ChildClass.class;
    System.out.println("类的名称:" + mClass.getName());
    System.out.println("----获取ChildClass类的私有方法privatePrintOneMsg---");
    // 2. 获取对应的私有方法
    // 第一个参数为要获取的私有方法的名称
    // 第二个为要获取方法的参数的类型,参数为 Class...,没有参数就是null
    // 方法参数也可这么写 :new Class[]{String.class}
    Method privateMethod = mClass.getDeclaredMethod("privatePrintOneMsg", String.class);
    // 3. 开始操作方法
    if (privateMethod != null) {
      // 获取私有方法的访问权
      // 只是获取访问权,并不是修改实际权限
      privateMethod.setAccessible(true);
      // 实例化对象
      ChildClass obj = (ChildClass) mClass.newInstance();
      // 使用 invoke 反射调用私有方法
      // obj 要操作的对象
      // 后面参数传实参
      privateMethod.invoke(obj, "hello world");
    }
  }
}

输出结果:

类的名称:com.example.java.reflect.ChildClass
----获取ChildClass类的私有方法privatePrintOneMsg---
子类-私有方法,变量:hello world

四、总结

由于反射会额外消耗一定的系统资源,因此如果不需要动态地创建一个对象,那么就不需要用反射。另外,反射调用方法时可以忽略权限检查,因此可能会破坏封装性而导致安全问题。

五、参考文章

sczyh30:深入解析Java反射

伯特:Java 反射由浅入深

相关文章
|
23小时前
|
安全 Java 测试技术
Java中的反射与元编程技术探秘
Java中的反射与元编程技术探秘
|
1天前
|
自然语言处理 Java 测试技术
Java中的行为驱动开发(BDD)方法论解析
Java中的行为驱动开发(BDD)方法论解析
|
1天前
|
安全 Java 数据安全/隐私保护
JAVA反射:你的代码“活”了吗?
【7月更文挑战第1天】Java反射是运行时动态代码操作,允许访问和修改类、属性及方法。它提供动态加载类、按需调用方法、修改私有属性及创建泛型实例的能力。使用反射需注意性能损耗、安全风险和代码可读性降低。示例展示了通过`Class.forName()`加载类,`getDeclaredConstructor().newInstance()`创建对象,`getMethod().invoke()`调用方法的过程。在性能敏感和安全关键的项目中使用时需谨慎。
|
1天前
|
安全 Java
JAVA反射:如何在运行时“创造”你的对象?
【7月更文挑战第1天】Java反射机制允许在运行时动态创建对象,常用于插件系统和框架开发。与`new`关键字相比,反射提供更大的灵活性,能根据条件决定对象类型,但牺牲了性能和代码可读性。示例展示了如何通过`Class.forName()`和`Constructor.newInstance()`创建对象,过程中需处理异常。反射还可访问私有成员,增加安全风险,故使用时需谨慎。
8 2
|
1天前
|
安全 Java API
JAVA反射:不只是“看”,还能“动手”改变!
【7月更文挑战第1天】Java反射提供运行时类信息查看与动态操作:获取类名、创建对象、调用未知方法、修改字段。虽强大但有性能和安全风险,需谨慎使用。
|
1天前
|
消息中间件 Java API
解析Java微服务架构:从零构建高性能系统
解析Java微服务架构:从零构建高性能系统
|
1天前
|
XML 缓存 JavaScript
优化Java中的XML解析性能
优化Java中的XML解析性能
|
1天前
|
Java
Java中多态的实现原理解析
Java中多态的实现原理解析
|
1天前
|
Java
解析Java中的反射机制应用
解析Java中的反射机制应用
|
1天前
|
设计模式 缓存 Java
Java中的反射机制:使用场景与注意事项
Java中的反射机制:使用场景与注意事项

推荐镜像

更多