反射的概述
Java反射机制的概念
Java反射机制是指在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态盗用对象方法的功能成为java语音的反射机制。
Java反射机制的作用
是用来编写一些通用性较高的代码或者框架的时候使用
反射常用对象的概述
Class:Class类的实例表示正在运行的Java应用程序中的类和接口
Constructor:关于类的单个构造方法的信息以及对它的访问权限
Field:Field提供有关类或接口的单个字段的信息,以及对它的动态访问权限
Method:Method提供关于类或接口上单独某个方法的信息
反射常用API
Class类
- Java中java.lang.Class类用于表示一个类的字节码(.class)文件
- 如何得到某个class文件对应的Class对象
已知类和对象的情况下:类名.class,对象.getClass() —- Object类提供
未知类和对象的情况下:Class.forName(“包名.类名”)
Class类代表某个类的字节码,并提供了加载字节码的方法:forName(“包名.类名”)
forName方法用于加载类字节码到内存中,并封装成一个Class对象
package com.song.reflect;
import org.junit.Test;
public class ClassTest {
@Test
public void demo1() throws ClassNotFoundException {
// 1. 通过类名.class的方式
Class class1 = Person.class;
System.out.println("class1:" + class1);
// 2. 通过对象.getClass()的方式
Person person = new Person();
Class class2 = person.getClass();
System.out.println("class2:" + class1);
// 3. 通过Class.forName()的方式(推荐)
Class class3 = Class.forName("com.song.reflect.Person");
System.out.println("class3:" + class1);
}
}
class1:class com.song.reflect.Person
class2:class com.song.reflect.Person
class3:class com.song.reflect.Person
Constructor类
- Constructor类的实例对象代表类的一个构造方法
- 得到某个类所有的构造方法
Constructor[] constructors = Class.forName(“java.lang.String”).getConstructors();
- 得到指定的构造方法并调用
Constructor constructor = Class.forName(“java.lang.String”).getConstructor(String.class);
String str = (String)constructor.newInstance(“abc”);
- Class类的newInstance()方法用来调用类的默认构造方法
String obj = (String)Class.forName(“java.lang.String”).newInstance();
package com.song.reflect;
import org.junit.Test;
import java.lang.reflect.Constructor;
public class ConstructorTest {
@Test
// 获得无参数的构造方法
public void demo1() throws Exception {
Class class1 = Class.forName("com.song.reflect.Person");
Constructor constructor = class1.getConstructor();
Person person = (Person)constructor.newInstance(); // 相当于Person person = new Person();
person.eat(); // eat...
System.out.println(person); // Person{name='null', sex='null'}
}
@Test
// 获得有参数的构造方法
public void demo2() throws Exception {
Class class1 = Class.forName("com.song.reflect.Person");
Constructor constructor = class1.getConstructor(String.class, String.class);
Person person = (Person)constructor.newInstance("张三","男"); // 相当于Person person = new Person("张三","男");
// Person person = (Person)constructor.newInstance(); => 注意这样是会编译出错的,因为得到的构造方法是带参数的
person.eat(); // eat...
System.out.println(person); // Person{name='张三', sex='男'}
}
}
Field类
- Field类代表某个类中的一个成员变量,并提供动态的访问权限
- Field对象的而获得
得到所有的成员变量
Field[] fields = c.getFields(); // 取得所有的public属性(包括父类继承)
Fiels[] fields = c.getDeclaredFields(); // 取得所有声明的属性
得到指定的成员变量
Field name = c.getField(“name”);
Field name = c.getDeclaredField(“name”);
- 设置Field变量是否可以访问
field.setAccessible(boolean);
- Field变量值的读取,设置
field.get(obj)
field.set(obj,value)
package com.song.reflect;
import org.junit.Test;
import java.lang.reflect.Field;
public class FieldTest {
@Test
// 测试共有的属性
public void demo1() throws Exception {
// 获得Class
Class class1 = Class.forName("com.song.reflect.Person");
// 获得属性
Field field = class1.getField("name");
// 操作属性
Person person = (Person) class1.newInstance();
field.set(person, "李四");
System.out.println(person); // Person{name='李四', sex='null'}
}
@Test
// 测试私有属性
public void demo2() throws Exception {
// 获得Class
Class class1 = Class.forName("com.song.reflect.Person");
// 获得属性
Field field = class1.getDeclaredField("sex");
// 操作属性
Person person = (Person) class1.newInstance();
field.setAccessible(true); // 私有属性,需要设置一个可访问的权限
field.set(person, "男");
System.out.println(person); // Person{name='null', sex='男'}
}
}
Method类
- Method类代表某个类中的一个成员方法
- Method对象的获得
获得所有方法
getDeclaredMethods()
getMethods()
获得指定的方法
getDeclaredMethod(String name, Class<?>…parameterTypes)
getMethod(String name, Class<?>…parameterTypes)
- 通过反射执行方法
invoke(Object obj, Object… args)
package com.song.reflect;
import org.junit.Test;
import java.lang.reflect.Method;
public class MethodTest {
@Test
// 测试共有的方法
public void demo1() throws Exception {
// 获得Class
Class class1 = Class.forName("com.song.reflect.Person");
// 获得属性
Method method = class1.getMethod("eat");
// 执行方法
Person person = (Person) class1.newInstance();
method.invoke(person); // eat...
}
@Test
// 测试私有的方法
public void demo2() throws Exception {
// 获得Class
Class class1 = Class.forName("com.song.reflect.Person");
// 获得属性
Method method = class1.getDeclaredMethod("run");
// 执行方法
Person person = (Person) class1.newInstance();
method.setAccessible(true); // 私有方法,需要设置一个可访问的权限
method.invoke(person); // eat...
}
@Test
// 测试带参数的方法
public void demo3() throws Exception {
// 获得Class
Class class1 = Class.forName("com.song.reflect.Person");
// 获得属性
Method method = class1.getDeclaredMethod("sayHello", String.class);
// 执行方法
Person person = (Person) class1.newInstance();
method.setAccessible(true); // 私有方法,需要设置一个可访问的权限
Object obj = method.invoke(person, "Song"); // eat...
System.out.println(obj);
}
}