对于一般的开发者,很少需要直接使用Java反射机制来完成功能开发,但是反射是很多框架譬如 Spring, Mybatis 实现的核心,反射虽小,能量却很大。
本文主要介绍反射相关的概念以及API的使用,关于反射的应用将在下一篇文章中介绍
反射的介绍
反射(Reflection) 是 Java 在运行时(Run time)可以访问、检测和修改它本身状态或行为的一种能力,它允许运行中的 Java 程序获取自身的信息,并且可以操作类或对象的内部属性。
Class 类介绍:Java虚拟机为每个类型管理一个Class对象,包含了与类有关的信息,当通过 javac 编译Java类文件时,生成的同名 .class 文件保存着该类的 Class 对象,JVM 加载一个类即是加载该 .class 文件。
Class 和 java.lang.reflect 一起对反射提供了支持,java.lang.reflect 包中最常用的几个类的关系如下:
其中最主要的三个类 Field、 Method 和 Constructor 分别用于描述类的域、方法和构造器,它们有一个共同的父类 AccessibleObject,它提供了访问控制检查的功能。
Field :描述类的域(属性),可以使用 get() 和 set() 方法读取和修改 Field 对象关联的字段;
Method :描述类的方法,可以使用 invoke() 方法调用与 Method 对象关联的方法;
Constructor :描述类的构造器,可以用 Constructor 创建新的对象。
下面将通过几个程序来学习Java反射机制。
准备两个类用于实验
我们特别定义两个类,Person和Employee,其中Employee继承自Person,且各自都有一个private,protected,public修饰的域(属性),Employee还有private,public修饰的方法
public class Person { public String name; // 姓名 公有 protected String age; // 年龄 保护 private String hobby; // 爱好 私有 public Person(String name, String age, String hobby) { this.name = name; this.age = age; this.hobby = hobby; } public String getHobby() { return hobby; }}public class Employee extends Person { public static Integer totalNum = 0; // 员工数 public int empNo; // 员工编号 公有 protected String position; // 职位 保护 private int salary; // 工资 私有 public void sayHello() { System.out.println(String.format("Hello, 我是 %s, 今年 %s 岁, 爱好是%s, 我目前的工作是%s, 月入%s元\n", name, age, getHobby(), position, salary)); } private void work() { System.out.println(String.format("My name is %s, 工作中勿扰.", name)); } public Employee(String name, String age, String hobby, int empNo, String position, int salary) { super(name, age, hobby); this.empNo = empNo; this.position = position; this.salary = salary; Employee.totalNum++; }}复制代码
获取 Class 对象
获取 Class 对象的方式有三种:使用 Class 类的 forName 静态方法;直接获取某一个对象的 class;调用某个对象的 getClass() 方法
public class ClassTest { public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException { Class c1 = Class.forName("reflect.Employee"); // 第1种,forName 方式获取Class对象 Class c2 = Employee.class; // 第2种,直接通过类获取Class对象 Employee employee = new Employee("小明", "18", "写代码", 1, "Java攻城狮", 100000); Class c3 = employee.getClass(); // 第3种,通过调用对象的getClass()方法获取Class对象 if (c1 == c2 && c1 == c3) { // 可以通过 == 比较Class对象是否为同一个对象 System.out.println("c1、c2、c3 为同一个对象"); System.out.println(c1); // class reflect.Employee } }}复制代码
通过反射来创建实例
通过反射来生成对象主要有两种方式
使用Class对象的newInstance()方法来创建Class对象对应类的实例
先通过Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建实例
public class NewInstanceTest { public static void main(String[] args) throws IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException { Class c = Date.class; Date date