【Java基础】Java中的反射机制

一、反射的理解

(1)正射

在理解反射这个概念之前,我们先来理解Java中的“正射”。

我们在编写代码时,当需要使用到某一个类的时候,必定先会去了解这是一个什么类,是用来做什么的,有怎么样的功能。

之后我们才对这个类进行实例化,之后再使用这个类的实例化对象进行操作。

Person person = new Person();
person.sleep("8:00");

(2)反射

上面的栗子介绍了什么是“正射”,以及“正射”的一般代码实现;

而反射则是在代码一开始编写时不知道要初始化的类是什么。因此,自然也无法使用new关键字来创建对象了。

而当我们之后得到我们要初始化的类的名称及路径时,我们就可以使用JDK提供的反射API进行反射调用。

Class clazz = Class.forName("com.test.domain.Person");
Method method = clazz.getMethod("sleep", String.class);
Constructor constructor = clazz.getConstructor();
Object object = constructor.newInstance();
method.invoke(object, "8:00");

以上两段代码,其结果都是一样的,但是其实现的过程却有很大的差别:

  • 第一段代码在未运行前就已经确定了要运行的类(Person);
  • 第二段代码则是在整个程序运行时从某些地方(例:配置文件)获取到相应的字符串值才能知道要运行的类("com.test.domain.Person")。

 

二、反射的应用

在我们日常的生产环境中,很少会直接使用到反射,但这并不代表反射在实际应用中很少。相反反射在Java中的框架使用得十分的多,反射是框架设计的灵魂:

①:spring、hibernate中会使用到反射机制,最常见的就是使用XML配置文件获取对应的类,然后再加载;

②:真的好多啊。。。

三、反射的常用函数

(1)获取反射中的class对象

在反射中,要获取一个类或调用一个类的方法,首先必须要获取到该类的对象,在Java API中,获取Class类对象三种方法:

①:Class.forName("类的路径名");

Class clazz = Class.forName("com.test.domain.Person");

②:利用已有类对象的getClass()方法;

Person person = new Person();
Class clazz = person.getClass();

③:对于在编译前就已经知道的类,可以使用.class属性;

Class clazz = Person.class;

(2)通过反射创建类对象

通过反射建立类的对象,Java API提供了两种方式:

①:通过class对象的newInstance()方法;

Class clazz = Class.forName("com.test.domain.Person"); 
Person person
= (Person)clazz.newInstance();

②:通过Constructor对象的newInstance()方法;

Class clazz = Class.forName("com.test.domain.Person");
Constructor con = clazz.getConstructor();
Person person = (Person)con.newInstance();

(3)通过反射操作成员变量

①:获取所有成员getFields()&getDeclaredFields();

使用getFields()方法可以获取Class类的成员变量,但是无法获取私有属性。

Class clazz = Class.forName("com.test.domain.Person");
Field[] fields = clazz.getFields();
for (Field field : fields) {
  System.out.print(field.getName());
}

②:获取单个成员getField()&getDeclared

③:修改成员变量的值set(Object obj, Object value)

Class clazz = Class.forName("com.test.domain.Person");
Person person = (Person)clazz.newInstance();
Field field = clazz.getField("name");
field.set(person, "张三");

当属性为private时,这是我们无法直接使用set()方法修改它的值,此时应该使用setAccessible()方法取得访问权限:

Class clazz = Class.forName("com.test.domain.Person");
Person person = (Person)clazz.newInstance();
Field field = clazz.getDeclaredField("name");
field.setAccessible(true);
field.set(person, "张三");

(4)通过反射操作成员方法

①:获取所有方法getMethods()&getDeclaredMethods()

②:获取单个成员getMethod()&getDeclaredMethod()

操作方法与操作变量相差不大,在获取到对应方法之后使用invoke()方法执行即可。同理,遇见私有方法时,也需要使用setAccessible(true)方法获取访问权限。

Class clazz = Class.forName("com.test.domain.Person");
Person person = (Person)clazz.newInstance();
Method method = clazz.getMethod("setName", String.class);
method.setAccessible(true);
method.invoke("李四");

通常情况下,即便得到的是当前类,private修饰的属性或方法也是没有权限访问的,你需要设置访问权限setAccessible(true)来取得访问权限,但在实际上,这已经破坏了规则,所以应该尽量少地使用。

posted @ 2019-04-08 22:17  周二鸭  阅读(1327)  评论(0编辑  收藏  举报