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 - 参数列表。

相关文章
|
4月前
|
前端开发 JavaScript Java
Java 开发中 Swing 界面嵌入浏览器实现方法详解
摘要:Java中嵌入浏览器可通过多种技术实现:1) JCEF框架利用Chromium内核,适合复杂网页;2) JEditorPane组件支持简单HTML显示,但功能有限;3) DJNativeSwing-SWT可内嵌浏览器,需特定内核支持;4) JavaFX WebView结合Swing可完美支持现代网页技术。每种方案各有特点,开发者需根据项目需求选择合适方法,如JCEF适合高性能要求,JEditorPane适合简单展示。(149字)
466 1
|
29天前
|
Java
Java语言实现字母大小写转换的方法
Java提供了多种灵活的方法来处理字符串中的字母大小写转换。根据具体需求,可以选择适合的方法来实现。在大多数情况下,使用 String类或 Character类的方法已经足够。但是,在需要更复杂的逻辑或处理非常规字符集时,可以通过字符流或手动遍历字符串来实现更精细的控制。
202 18
|
1月前
|
Java 编译器 Go
【Java】(5)方法的概念、方法的调用、方法重载、构造方法的创建
Java方法是语句的集合,它们在一起执行一个功能。方法是解决一类问题的步骤的有序组合方法包含于类或对象中方法在程序中被创建,在其他地方被引用方法的优点使程序变得更简短而清晰。有利于程序维护。可以提高程序开发的效率。提高了代码的重用性。方法的名字的第一个单词应以小写字母作为开头,后面的单词则用大写字母开头写,不使用连接符。例如:addPerson。这种就属于驼峰写法下划线可能出现在 JUnit 测试方法名称中用以分隔名称的逻辑组件。
168 4
|
2月前
|
算法 安全 Java
除了类,Java中的接口和方法也可以使用泛型吗?
除了类,Java中的接口和方法也可以使用泛型吗?
122 11
|
1月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
223 5
|
2月前
|
Java 开发者
Java 函数式编程全解析:静态方法引用、实例方法引用、特定类型方法引用与构造器引用实战教程
本文介绍Java 8函数式编程中的四种方法引用:静态、实例、特定类型及构造器引用,通过简洁示例演示其用法,帮助开发者提升代码可读性与简洁性。
|
3月前
|
算法 Java 开发者
Java 项目实战数字华容道与石头迷阵游戏开发详解及实战方法
本文介绍了使用Java实现数字华容道和石头迷阵游戏的技术方案与应用实例,涵盖GUI界面设计、二维数组操作、游戏逻辑控制及自动解法算法(如A*),适合Java开发者学习游戏开发技巧。
251 46
|
4月前
|
Java 索引
Java ArrayList中的常见删除操作及方法详解。
通过这些方法,Java `ArrayList` 提供了灵活而强大的操作来处理元素的移除,这些方法能够满足不同场景下的需求。
483 30
|
3月前
|
算法 Java
Java语言实现链表反转的方法
这种反转方法不需要使用额外的存储空间,因此空间复杂度为,它只需要遍历一次链表,所以时间复杂度为,其中为链表的长度。这使得这种反转链表的方法既高效又实用。
334 0
|
4月前
|
安全 Java API
Java 17 及以上版本核心特性在现代开发实践中的深度应用与高效实践方法 Java 开发实践
本项目以“学生成绩管理系统”为例,深入实践Java 17+核心特性与现代开发技术。采用Spring Boot 3.1、WebFlux、R2DBC等构建响应式应用,结合Record类、模式匹配、Stream优化等新特性提升代码质量。涵盖容器化部署(Docker)、自动化测试、性能优化及安全加固,全面展示Java最新技术在实际项目中的应用,助力开发者掌握现代化Java开发方法。
185 1