首页 > 编程语言 >Java通过反射生成并操作对象

Java通过反射生成并操作对象

时间:2022-11-20 19:14:31浏览次数:64  
标签:反射 Java 对象 程序 生成 Field 数组 方法

Java通过反射生成并操作对象

1.* 使用反射生成并操作对象

Class对象可以获得该类里的方法(由Method对象表示)、构造器(由Constructor对象表示)、Field(由Field对象表示),这3个类都位于java.lang.reflect包下,并实现了java.lang.reflect.Member接口。程序可以通过Method对象来执行对应的方法,通过Constructor对象来调用对应的构造器创建实例,能通过Field对象直接访问并修改对象的属性值。

1.*.& 创建对象

通过反射来生成对象有如下两种方式:

  • 使用Class对象的newInstance()方法来创建该Class对象对应类的实例,这种方式要求该Class对象的对应类有默认构造器,而执行newInstance()方法时实际上是利用默认构造器来创建该类的实例。
  • 先使用Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建该Class对象对应类的实例。通过这种方式可以选择使用指定的构造器来创建实例。

通过第一种方式来创建对象是比较常见的情形,因为在很多JavaEE框架中都需要根据配置文件信息来创建Java对象,从配置文件读取的只是某个类的字符串类名,程序需要根据该字符串来创建对应的实例,就必须使用反射。

下面程序就实现了一个简单的对象池,该对象池会根据配置文件读取key-value对,然后创建这些对象,并将这些对象放入一个HashMap中:

查看代码

上面程序中createObject()方法里的代码就是根据字符串来创建Java对象的关键代码,程序调用Class对象的newInstance()方法即可创建一个Java对象。程序中的initPool()方法会读取属性文件,对属性文件中每个key-value对创建一个Java对象,其中value是该Java对象的实现类,而key是该Java对象放入对象池中的名字。为该程序提供如下属性配置文件:

点击查看代码
a=java.util.Date
b=javax.swing.JFrame

编译、运行上面的ObjectPoolFactory程序,执行到main方法中的[1]号代码处,将看到输出系统当前时间,这表明对象池中已经有了一个名为a的对象,该对象是一个java.util.Date对象。执行到[2]号代码处,将看到输出一个JFrame对象。

提示:这种使用配置文件来配置对象,然后由程序根据配置文件来创建对象的方式非常有用,大名鼎鼎的Spring框架就采用这种方式大大简化了JavaEE应用的开发。当然,Spring采用的是XML配置文件,毕竟属性文件能配置的信息太有限了,而XML配置文件能配置的信息就丰富多了。

如果不想利用默认构造器来创建Java对象,而想利用指定的构造器来创建Java对象,则需要利用Constructor对象,每个Constructor对应一个构造器。为了利用指定的构造器来创建Java对象,需要如下3个步骤:

1.获取该类的Class对象。

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

3.调用Constructor的newInstance()方法来创建Java对象。

下面程序利用反射来创建一个JFrame对象,而且使用指定的构造器:

查看代码

上面程序中代码先是获取JFrame类的指定构造器,前面已经提到:如果要唯一地确定某类中的构造器,只要指定构造器的形参列表即可。Constructor ctor = jframeClazz.getConstructor(String.class)获取构造器时传入了一个String类型,即表明想获取只有一个字符串参数的构造器。

然后使用指定构造器的newInstance()方法来创建一个Java对象,当调用Constructor对象的newInstance()方法时通常需要传入参数,因为调用Constructor的newInstance()方法实际上等于调用它对应的构造器,传给newInstance()方法的参数将作为对应构造器的参数。

对于上面的CreateFrame.java中已知java.swing.JFrame类的情形,通常没有必要使用反射来创建该对象,毕竟通过反射创建对象时性能要稍低一些。实际上,只有当程序需要动态创建某个类的对象时才会考虑使用反射,通常在开发通用性比较广的框架、基础平台时可能会大量使用反射。

1.*.& 调用方法

当获得某个类对应的Class对象后,就可以通过该Class对象的getMethods()方法或者getMethod()方法来获取全部方法或指定方法,这两个方法的返回值是Method数组,或者Method对象。

每个Method对象对应一个方法,获得Method对象后,程序就可通过该Method来调用它对应的方法。在Method里包含一个invoke()方法,该方法的签名如下:

  • Object invoke(Object obj, Object...args):该方法中的obj是执行该方法的主调,后面的args是执行该方法时传入该方法的实参。

下面程序对前面的对象池工厂进行加强,允许在配置文件中增加配置对象的属性值,对象池工厂会读取该对象的属性值,并利用该对象对应的setter方法为对应属性设置值:

查看代码

上面程序中initProperty()方法里的获取目标类中包含一个String参数的setter方法,然后通过调用Method的invoke()方法来执行该setter方法,该方法执行完成后,就相当于执行了目标对象的setter方法。为上面程序提供如下配置文件:

点击查看代码
a=javax.swing.JFrame
b=javax.swing.JLabel
a%title=Test Title

上面配置文件中的a%title=Test Title行表明为a对象的title字段设置值。编译、运行上面的ExtendedObjectPoolFactory.java程序,可以看到输出一个JFrame窗口,该窗口的标题属性为Test Title

提示:Spring框架就是通过这种将Field值以及依赖对象等都放在配置文件中进行管理的,从而实现了较好的解耦。这也是Spring框架的IoC的秘密。

当通过Method的invoke()方法来调用对应的方法时,Java会要求程序必须有调用该方法的权限。如果程序确实需要调用某个对象的private方法,则可以先调用Method对象的如下方法:

  • setAccessible(boolean flag):将Method对象的accessible设置为指定的布尔值。值为true,指示该Method在使用时应该取消Java语言的访问权限检查;值为false,则指示该Method在使用时要实施Java语言的访问权限检查。

注意:实际上,setAccessible()方法并不属于Method,而是属于它的父类AccessibleObject。因此Method、Constructor、Field都可调用该方法,从而实现通过反射来调用private方法,private属性和private构造器。也就是说它们可以通过调用该方法来取消访问权限检查,通过反射即可访问private的成员。

1.*.& 访问属性值

通过Class对象的getFields()或getField()方法可以获取该类所包括的全部Field或指定Field。Field提供了如下两组方法来读取或设置Field值:

  • getXxx(Object obj):获取obj对象该Field的属性值。此处的Xxx对应8个基本类型,如果该属性的类型是引用类型,则取消get后面的Xxx。
  • setXxx(Object obj,Xxx val):将obj对象的该Field设置成val值。此处的Xxx对应8个基本类型,如果该属性的类型是引用类型,则取消set后面的Xxx。

使用这两个方法可以随意地访问指定对象的所有属性,包括private访问控制的属性。

如下代码:

查看代码

上面程序中先定义了一个Person类,该类里包含两个private Field:name和age,在通常情况下,这两个Field只能在Person类里访问。但本程序FieldTest的main方法中通过反射修改了Person对象的name、age两个Field的值。

上面的程序首先使用getDeclaredField()方法获取了名为name的Field,注意此处不是使用getField()方法,因为getField()方法只能获取public访问控制的Field,而getDeclaredField()方法则可以获取所有的Field;然后通过反射访问该Field时不受访问权限的控制;接着修改了Person对象的name属性值。修改Person对象的age属性值的方式与此完全相同。

编译、运行上面程序,会看到如下输出:

点击查看代码
Person[name: Yeeku.H.Lee, age: 30]

1.*.& 操作数组

在java.lang.reflect包下还提供了一个Array类,Array对象可以代表所有的数组。程序可以通过使用Array来动态地创建数组,操作数组元素等。

Array提供了如下几类方法:

  • static Object newInstance(Class<?>componentType, int...length):创建一个具有指定的元素类型、指定维度的新数组。
  • static xxx getXxx(Object array, int index):返回array数组中第index个元素。其中xxx是各种基本数据类型,如果数组元素是引用类型,则该方法变为get(Object array, int index)。
  • static void setXxx(Object array, int index, xxx val):将array数组中第index个元素的值设为val。其中xxx是各种基本数据类型,如果数组元素是引用类型,则该方法变成set(Object array, int index, Object val)。

下面程序示范了如何使用Array来生成数组,为指定数组元素赋值,并获取指定数组元素的方式:

查看代码

上面程序中分别是通过Array创建数组,为数组元素设置值,访问数组元素的值的示例代码,程序通过使用Array就可以动态地创建并操作数组。

下面程序比上面程序稍微复杂一点,下面程序使用Array类创建了一个三维数组:

查看代码

上面程序先是使用Array创建了一个三维数组,程序中较难理解的地方是Array.set(arrObj, 2, new String[]{"疯狂Java讲义", "轻量级JavaEE企业应用实战"})部分,使用Array为arrObj的指定元素赋值,相当于为二维数组的元素赋值。我们知道二维数组的元素是一维数组,所以程序传入的参数是一个一维数组对象。

运行上面程序,将看到cast[2][3][8]、cast[2][2][0]、cast[2][2][1]元素都有值,这些值就是刚才程序通过反射传入的数组元素值。

标签:反射,Java,对象,程序,生成,Field,数组,方法
From: https://www.cnblogs.com/hzhiping/p/16908291.html

相关文章

  • Java通过反射查看类信息
    Java通过反射查看类信息1.*通过反射查看类信息Java程序中的许多对象在运行时都会出现两种类型:编译时类型和运行时类型,例如代码:“Personp=newStudent()”,这行代码将......
  • java——线程同步机制——解决线程安全问题——同步代码块
                                      解决线程安全问题——同步代码块卖票案例出现了线程安......
  • Linux下安装Java运行环境
    1.下载java8的包,并上传到服务器/usr/local目录下wget命令是一个从网络上下载文件的自由工具,它支持http协议,https协议和ftp协议。因此我们可以通过wget命令来下载JDK。wge......
  • java注解详解以及如何获取注解的上的信息
    目录一、Java自定义注解详解1.定义注解:2.元注解介绍@Target详细介绍@Relation详细介绍@Documented介绍@Inherited介绍3.注解可用的类型4.默认值限制5.创建一个简单的自定义......
  • Java中使用脚本引擎运行脚本语言
    在Java中运行脚本语言,例如JavaScript。步骤:1、创建脚本引擎管理器ScriptEngineManager2、从管理器中获取一个引擎ScriptEngine3、通过put(key,valu......
  • JAVA接口
    JDK1.8之前接口是接口,类是类。它们是同一层次的概念。接口中没有构造器。接口如何声明:interface在JDK1.8之前,接口中只有两部分内容,(1)常量:固定修饰符:publicstaticfina......
  • JavaWeb笔记
    1.JavaEE项目的三层架构web层                    com.atguigu.web/servlet/controllerservice层         ......
  • jenkins配置从节点后运行报错java.net.ConnectException: Connection timed out: conn
    修改jenkins配置中的ip系统管理-系统配置  修改ip与访问地址相同 ......
  • JAVA学习方法与知识点
       这个时代有很多的朋友都开始选择看看学习学习当下热门的编程语言比如现在的Java这类技术。俗话说的好啊天下熙熙皆为利来,天下攘攘皆为利往,目前大多都是为了高薪工......
  • javascript: 用图片加载演示promise的应用(chrome 107.0.5304.110)
    一,js代码:<html><head><metacharset="utf-8"/><title>测试</title></head><body><imgid="img"src=""/><script>//记录开始时间leta=newDate......