JAVA反射调用方法

简介: JAVA反射调用方法
一、利用反射创建对象

创建对象:

1、使用 Class 对象的 newInstance()方法创建该 Class 对象的实例,此时该 Class 对象必须要有无参数的构造方法。

2、使用 Class 对象获取指定的 Constructor 对象,再调用 Constructor 的 newInstance()方法创建对象类的实例,此时可以选择使用某个构造方法。如果这个构造方法被私有化起来,那么必须先申请访问,将可以访问设置为 true;

示例代码(最简单的)

class User {  
  /*private User(){//将默认的构造方法私有化的话就不可以再创建对象,两种方法都是这样    
  }*/  
 
  public String toString() {
    return "User对象创建成功!";
  }
}
 
public class ReflectDemo {
  public static void main(String[] args) throws Exception {
    // 传统方式创建对象
    System.out.println(new User());
    // 使用反射的方式
    Class<User> c = User.class;
    User u = c.newInstance();// (直接newInstance的话必须保证默认的构造方法正常存在,也就是没有被私有化!这是前提条件)
    System.out.println(u);
  }
}

结果截图如下:

复杂点的,更强大的第二种:使用指定构造方法来创建对象:获取该类的 Class 对象。

利用 Class 对象的 getConstructor()方法来获取指定的构造方法。

调用 Constructor 的 newInstance()方法创建对象。

AccessibleObject 对象的 setAccessible(boolean flag)方法,当 flag 为 true 的时候,就会忽略访问

权限(可访问私有的成员)。

其子类有 Field, Method, Constructor;

若要访问对象 private 的成员?在调用之前使用 setAccessible(true),参考http://www.wityx.com/post/1202_1_1.html

Xxx x = getDeclaredXxxx();//才能得到私有的类字段.

总结步骤:
  1. 获取该类的 Class 对象。
  2. 利用 Class 对象的 getConstructor()方法来获取指定的构造方法。
  3. 申请访问(设置为可访问)
  4. 调用 Constructor(构造方法)的 newInstance()方法创建对象。

示例代码:

import java.lang.reflect.Constructor;
//http://www.wityx.com/post/256_1_1.html
class Per {
  private String name;
  private int age;
 
  private Per() {
  }
 
  private Per(String name) {
  }
 
  public String toString() {
    return "对象!!!";
  }
}
 
public class ReflectDemo1 {
  public static void main(String[] args) throws Exception {
    Class<Per> c = Per.class;
    // System.out.println(c.newInstance());;//证明利用无参的可以
    // 先获得需要被调用的构造器(private 修饰的构造方法)
    Constructor<Per> con = c.getDeclaredConstructor();// 调用默认的,什么都不要写
    System.out.println(con);    
    /**
     * 现在只需要执行这个构造器,T newInstance(Object... initargs) 使用此 Constructor
     * 对象表示的构造方法来创建该构造方法的声明 类的新实例,并用指定的初始化参数初始化该实例。
     */
    // 私有的成员是受保护的,不能直接访问(无法直接访问的构造器不一定是私有的),若要访问私有的成员,得先申请一下
    //http://www.wityx.com/post/935_1_1.html
    con.setAccessible(true);// 允许访问
    Per p = con.newInstance();// 成功,通过私有的受保护的构造方法创建了对象
    System.out.println("无参构造方法" + p);
    con = c.getDeclaredConstructor(String.class);//获取指定的构造方法
    System.out.println(con);
    con.setAccessible(true);// 允许访问
    p = con.newInstance("duixiang");// 成功,通过私有的受保护的构造方法创建了对象
    System.out.println("String构造方法" + p);
  }
}

备注:对于此时的话,单例模式就不再安全了!反射可破之!!

结果截图:

总结:

(1)getDeclaredConstructor()当没有参数时,它就是默认获取无参构造器,也可以获取指定的构造方法,如con = c.getDeclaredConstructor(String.class);//获取指定的构造方法

(2)T newInstance(Object… initargs) 使用此 Constructor 对象表示的构造方法来创建该构造方法的声明类的新实例,并用指定的初始化参数初始化该实例。

(3)私有的成员是受保护的,不能直接访问(无法直接访问的构造器不一定是私有的),若要访问私有的成员,得先申请一下,如con.setAccessible(true);// 允许访问

二、使用反射调用方法

每个 Method 的对象对应一个具体的底层方法。获得 Method 对象后,程序可以使用 Method 里面的 invoke 方法来执行该底层方法。

Object invoke(Object obj,Object … args):obj 表示调用底层方法的对象,后面的 args 表示传递

的实际参数。

如果底层方法是静态的,那么可以忽略指定的 obj 参数。该参数可以为 null,想想为什么?

如果底层方法所需的形参个数为 0,则所提供的 args 数组长度可以为 0 或 null。

不写,null,或 new Object[]{}

若底层方法返回的是数组类型,invoke 方法返回的不是底层方法的值,而是底层方法的返回类

型;

示例代码:

import java.lang.reflect.Method;
//http://www.wityx.com/post/640_1_1.html
class Dept {
  public String show(String name) {// 用反射的方法来调用正常的方法
    return name + ",您好!";
  }
 
  private void privateshow() {// 用反射来实现对私有化方法的调用
    System.out.println("私有的方法:privateshow");
  }
 
  public static void staticshow() {
    System.out.println("静态的方法:staticshow");
  }
}
 
public class InvokeDemo {
  public static void main(String[] args) throws Exception {
    //传统方式: 
    String name = new Dept().show("刘昭"); 
    System.out.println("传统方式:" + name);
    // 想要通过反射来调用Dept中的方法
    Class<Dept> c = Dept.class;
    Method m = c.getMethod("show", String.class);
    Object o = m.invoke(c.newInstance(), "刘昭");
    System.out.println("反射调用的方式:" + o);
    // 私有化的方法
    // 参考http://www.wityx.com/post/1205_1_1.html
    m = c.getDeclaredMethod("privateshow");// 无参方法
    m.setAccessible(true);
    o = m.invoke(c.newInstance());
    // 静态方法的调用
    m = c.getMethod("staticshow");
    m.invoke(null);// staticshow为静态方法,不需创建对象,所以这里会是 null
  }
}

结果截图:

注:Method getMethod(String name, Class<?>… parameterTypes)返回一个 Method 对象,它反映此 Class 对象所表示的类或接口的指定公共成员方法。name - 方法名;parameterTypes - 参数列表。

相关文章
|
7天前
|
Java
Java——方法的引用
方法引用允许将已有方法作为函数式接口的实现。使用“::”符号,需具备函数式接口,被引用的方法须存在且参数和返回值需与抽象方法一致。其分类包括:静态方法引用(类::方法名)、成员方法引用(对象::方法名、this::方法名、super::方法名)和构造方法引用(类名::new)。方法引用提高了代码的简洁性和可读性,减少了样板代码。
25 13
Java——方法的引用
|
7天前
|
安全 Java 索引
Java——反射&枚举
本文介绍了Java反射机制及其应用,包括获取Class对象、构造方法、成员变量和成员方法。反射允许在运行时动态操作类和对象,例如创建对象、调用方法和访问字段。文章详细解释了不同方法的使用方式及其注意事项,并展示了如何通过反射获取类的各种信息。此外,还介绍了枚举类型的特点和使用方法,包括枚举的构造方法及其在反射中的特殊处理。
30 9
Java——反射&枚举
|
3天前
|
Java
java基础(12)抽象类以及抽象方法abstract以及ArrayList对象使用
本文介绍了Java中抽象类和抽象方法的使用,以及ArrayList的基本操作,包括添加、获取、删除元素和判断列表是否为空。
9 2
java基础(12)抽象类以及抽象方法abstract以及ArrayList对象使用
|
14天前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
|
1天前
|
安全 Java API
Java根据URL获取文件内容的实现方法
此示例展示了如何安全、有效地根据URL获取文件内容。它不仅展现了处理网络资源的基本技巧,还体现了良好的异常处理实践。在实际开发中,根据项目需求,你可能还需要添加额外的功能,如设置连接超时、处理HTTP响应码等。
11 4
|
7天前
|
Java API
Java方法的优缺点
Java 方法是编程的基本构建块,具有代码重用性、模块化、易于调试、增强可读性、支持重载和可变参数、封装性及静态与实例方法的灵活性等优点,但也存在性能开销、过度抽象、限制使用环境、参数传递开销、命名冲突和堆栈溢出等缺点。合理设计方法可确保代码高效且易维护。
|
3天前
|
安全 Java
java调用方法
java调用方法
16 4
|
7天前
|
Java
Java的方法详解
在 Java 中,方法是执行特定任务的代码块,包括定义、参数传递、返回值处理及重载等功能。
|
16天前
|
Java
Java的方法详解
Java的方法是类中的重要组成部分,用于定义类的行为。方法可以接收参数、执行操作并返回结果。其基本语法包括返回类型、方法名、参数列表和方法体。方法支持重载,即同名但参数不同的多个方法;静态方法则直接通过类名调用,无需实例化。此外,Java还支持可变参数,允许方法接收不定数量的参数。通过访问修饰符如`public`、`protected`、`private`,可以控制方法的可见性。方法是实现类功能的基本单元,增强了程序的灵活性和复用性。
|
20天前
|
Java 程序员 编译器
Java的反射技术reflect
Java的反射技术允许程序在运行时动态加载和操作类,基于字节码文件构建中间语言代码,进而生成机器码在JVM上执行,实现了“一次编译,到处运行”。此技术虽需更多运行时间,但广泛应用于Spring框架的持续集成、动态配置及三大特性(IOC、DI、AOP)中,支持企业级应用的迭代升级和灵活配置管理,适用于集群部署与数据同步场景。