本文部分内容参考博客。点击链接可以查看原文。


1. 反射的概念

反射是指在运行时将类的属性、构造函数和方法等元素动态地映射成一个个对象。通过这些对象我们可以动态地生成对象实例,调用类的方法和更改类的属性值。

2. 使用场景

什么情况下运用JAVA反射呢?如果编译时根本无法预知对象和类可能属于哪些类,程序只依靠运行时信息来发现该对象和类的真实信息,此时就必须使用反射。

使用反射可以实现下面的功能:

  • 在运行时判断任意一个对象所属的类
  • 在运行时构造任意一个类的对象
  • 在运行时判断任意一个类所具有的方法和属性
  • 在运行时调用任意一个对象的方法
  • 生成动态代理

3. 获得Class对象的几种方式

前面已经介绍过了,每个类被加载之后,系统就会为该类生成一个对应的Class对象,通过该Class对象就可以访问到JVM中的这个类。在Java程序中获得Class对象通常有如下3种方式。

  • 使用Class类的forName(String clazzName)静态方法。该方法需要传入字符串参数,该字符串参数的值是某个类的全限定类名(必须添加完整包名)。
  • 调用某个类的class属性来获取该类对应的Class对象。例如,Person.class将会返回Person类对应的Class对象。
  • 调用某个对象的getClass()方法。该方法是java.lang.Object类中的一个方法,所以所有的Java对象都可以调用该方法,该方法将会返回该对象所属类对应的Class对象。

对于第一种方式和第二种方式都是直接根据类来取得该类的Class对象,相比之下,第二种方式有如下两种优势。

  • 代码更安全。程序在编译阶段就可以检查需要访问的Class对象是否存在。
  • 程序性能更好。因为这种方式无须调用方法,所以性能更好。

也就是说,大部分时候我们都应该使用第二种方式来获取指定类的Class对象。但如果我们只有一个字符串,例如“java.lang.String”,若需要获取该字符串对应的Class对象,则只能使用第一种方式,使用Class的forName(String clazzName)方法获取Class对象时,该方法可能抛出一个ClassNotFoundException异常。一旦获得了某个类所对应的Class对象之后,程序就可以调用Class对象的方法来获得该对象和该类的真实信息了。

4. Class类 API介绍

通过class类我们能够获取大量的信息:

  1. 获取构造函数
  • Connstructor getConstructor(Class<?>... parameterTypes):返回此Class对象对应类的指定public构造器。
  • Constructor<?>[] getConstructors():返回此Class对象对应类的所有public构造器。
  • Constructor getDeclaredConstructor(Class<?>... parameterTypes):返回此Class对象对应类的指定构造器,与构造器的访问权限无关。
  • Constructor<?>[] getDeclaredConstructors():返回此Class对象对应类的所有构造器,与构造器的访问权限无关。
  1. 获取方法
  • Method getDeclaredMethod(String name, Class<?>... parameterTypes):返回此Class对象对应类的指定方法,与方法的访问权限无关。
  • Method[] getDeclaredMethods():返回此Class对象对应类的全部方法,与方法的访问权限无关。
  1. 获取属性
  • Field getField(String name):返回此Class对象对应类的指定public Field。
  • Field[] getFields():返回此Class对象对应类的所有public Field。
  • Field getDeclaredField(String name):返回此Class对象对应类的指定Field,与Field的访问权限无关。
  • Field[] getDeclaredFields():返回此Class对象对应类的全部Field,与Field的访问权限无关。
  1. 获取Class对应类上所包含的Annotation。
  1. 获取Class对象对应类包含的内部类。
  • Class<?>[] getDeclaredClasses():返回该Class对象对应类里包含的全部内部类。

    如下方法用于访问该Class对象对应类所在的外部类。
  • Class<?> getDeclaringClass():返回该Class对象对应类所在的外部类。

    如下方法用于访问该Class对象对应类所继承的父类、所实现的接口等。
  • Class<?>[] getInterfaces():返回该Class对象对应类所实现的全部接口。
  1. 获取Class对象对应类所继承的父类
  • Class<? super T> getSuperclass():返回该Class对象对应类的超类的Class对象。
  1. 获取Class对象对应类的修饰符、所在包、类名等基本信息。
  • int getModifiers():返回此类或接口的所有修饰符。修饰符由public、protected、private、final、static、abstract等对应的常量组成,返回的整数应使用Modifier工具类的方法来解码,才可以获取真实的修饰符。
  • Package getPackage():获取此类的包。
  • String getName():以字符串形式返回此Class对象所表示的类的名称。
  • String getSimpleName():以字符串形式返回此Class对象所表示的类的简称。
  1. 判断该类是否为接口、枚举、注释类型等
  • boolean isAnnotation():返回此Class对象是否表示一个注释类型(由@interface定义)。
  • boolean isAnnotationPresent(Class<? extends Annotation> annotationClass):判断此Class对象是否使用了Annotation注释修饰。
  • boolean isAnonymousClass():返回此Class对象是否是一个匿名类。
  • boolean isArray():返回此Class对象是否表示一个数组类。
  • boolean isEnum():返回此Class对象是否表示一个枚举(由enum关键字定义)。
  • boolean isInterface():返回此Class对象是否表示一个接口(使用interface定义)。
  • boolean isInstance(Object obj):判断obj是否是此Class对象的实例,该方法可以完全代替instanceof操作符。

上面的多个getMethod()方法和getConstructor()方法中,都需要传入多个类型为Class<?>的参数,用于获取指定的方法或指定的构造器。关于这个参数的作用,假设某个类内包含如下3个info方法签名:

  • public void info()
  • public void info(String str)
  • public void info(String str , Integer num)

这3个同名方法属于重载,它们的方法名相同,但参数列表不同。在Java语言中要确定一个方法光有方法名是不行的,例如,我们指定info方法——实际上可以是上面3个方法中的任意一个!如果需要确定一个方法,则应该由方法名和形参列表来确定,但形参名没有任何实际意义,所以只能由形参类型来确定。例如,我们想要确定第二个info方法,则必须指定方法名为info,形参列表为String.class——因此在程序中获取该方法使用如下代码:

clazz.getMethod("info",String.class);

使用反射生成对象

  1. 利用构造函数生成对象
  • 使用Class对象的newInstance()方法来创建该Class对象对应类的实例,这种方式要求该Class对象的对应类有默认构造器,而执行newInstance()方法时实际上是利用默认构造器来创建该类的实例。
  • 先使用Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建该Class对象对应类的实例。通过这种方式可以选择使用指定的构造器来创建实例。
Constructor c = clazz.getConstructor(String.class);
c.newInstance("xx");

调用方法

Method setProName = aClass.getDeclaredMethod("setProName",String.class);
setProName.setAccessible(true);
etProName.invoke(product,"我是一个产品");

操作属性

Field[] declaredFields = aClass.getDeclaredFields();
for (Field declaredField : declaredFields) {
System.out.println("fieldName:"+declaredField.getName()+" filedType:"+declaredField.getType());
}
Field proName = aClass.getDeclaredField("proName");
proName.setAccessible(true);
proName.set(product,"我是一个产品");
System.out.println("修稿属性:"+product);

操作数组

//使用反射动态地创建数组
//创建一个元素类型为String,长度为3的数组
Object arr = Array.newInstance(String.class, 3);
//依次为arr数组中index为0,1,2的元素赋值
Array.set(arr, 0, "荣耀盒子");
Array.set(arr, 1, "荣耀8手机");
Array.set(arr, 2, "华为mate9保时捷版");
Object o1= Array.get(arr, 0);
Object o2= Array.get(arr, 1);
Object o3= Array.get(arr, 2);
System.out.println(o1);
System.out.println(o2);
System.out.println(o3);

5. 使用Demo

public class ReflectDemo {
public static void main(String[] args) throws Exception {
Class<Product> aClass = Product.class;
Constructor<?>[] declaredConstructors = aClass.getDeclaredConstructors();
for (Constructor<?> declaredConstructor : declaredConstructors) {
System.out.println(declaredConstructor.getName());
}
Constructor<Product> constructor = aClass.getConstructor(int.class, String.class);
Product product = constructor.newInstance(10, "ds");
System.out.println("创建对象:"+product); //获取方法并调用
Method setProName = aClass.getDeclaredMethod("setProName", String.class);
setProName.setAccessible(true);
setProName.invoke(product,"我是一个产品");
System.out.println("调用方法:"+product); //获取属性,并设置属性的值
Field[] declaredFields = aClass.getDeclaredFields();
for (Field declaredField : declaredFields) {
System.out.println("fieldName:"+declaredField.getName()+" filedType:"+declaredField.getType());
}
Field proName = aClass.getDeclaredField("proName");
proName.setAccessible(true);
proName.set(product,"我是一个产品");
System.out.println("修稿属性:"+product); //使用反射动态地创建数组
//创建一个元素类型为String,长度为3的数组
Object arr = Array.newInstance(String.class, 3);
//依次为arr数组中index为0,1,2的元素赋值
Array.set(arr, 0, "荣耀盒子");
Array.set(arr, 1, "荣耀8手机");
Array.set(arr, 2, "华为mate9保时捷版");
Object o1= Array.get(arr, 0);
Object o2= Array.get(arr, 1);
Object o3= Array.get(arr, 2);
System.out.println(o1);
System.out.println(o2);
System.out.println(o3); System.out.println("end...");
}
}

最新文章

  1. hdu1452 Happy 2004(规律+因子和+积性函数)
  2. Spark集群 + Akka + Kafka + Scala 开发(3) : 开发一个Akka + Spark的应用
  3. Change MYSQL data directory
  4. 【编程题目】一串首尾相连的珠子(m 个),有 N 种颜色(N&lt;=10),取出其中一段,要求包含所有 N 中颜色,并使长度最短。
  5. D3D标注动态避让
  6. WebRTC录音(2)-录音文件转换成WAV格式
  7. Target Operator ID has No Access to Upgrade
  8. OracleApps 什么是Back to Back Order?
  9. ios UIWebView截获html并修改便签内容(转载)
  10. error: ‘for’ loop initial declarations are only allowed in C99 mode
  11. Linux下实现C++类的动态链接
  12. Difference between Tomcat&#39;s extraResourcePaths and aliases to access an external directory--转
  13. 【Java】大文本字符串滤重的简单方案~
  14. Android学习记录:Paint,Canvas和Bitmap
  15. 51Nod.1766.树上最远点对(树的直径 RMQ 线段树/ST表)
  16. vuejs-devtools
  17. selenium批量翻译
  18. python中的运算符及表达式及常用内置函数
  19. kendo ui - DatePicker 日期时间系列
  20. 三维凸包求凸包表面的个数(HDU3662)

热门文章

  1. Java实现蓝桥杯 算法提高 盾神与积木游戏
  2. Java实现 蓝桥杯VIP 算法训练 装箱问题
  3. Java实现 蓝桥杯VIP 算法训练 ALGO-85进制转换
  4. MySQL死锁及解决方案
  5. cuda基础
  6. 曹工说JDK源码(1)--ConcurrentHashMap,扩容前大家同在一个哈希桶,为啥扩容后,你去新数组的高位,我只能去低位?
  7. 真香,撸一个SpringBoot在线代码修改器
  8. PAT1040 Longest Symmetric String (25分) 中心扩展法+动态规划
  9. HTTP协议浅析(一)
  10. 关于前端JS走马灯(marquee)总结