1.反射的定义
反射:(reflection):在运行时期,动态地去获取类中的信息(类的信息,方法信息,构造器信息,字段等信息进行操作)。
2.获取类的Class实例的三种方式
1. 类名.class
2. 类的对象.getClass()
3. Class.forName(“类的全限定名”) 全限定名 = 包名 + 类名
注意 :同一个类在JVM的字节码实例只有一份。
public class User {
@Test
public void testName() throws Exception {
//1.使用类名.class 获取类的字节码实例
Class<User> clz1 = User.class;
System.out.println(clz1.toString());
//2.对象.getClass()
User user = new User();
Class<?> clz2 = user.getClass();
System.out.println(clz2);
//3.Class.forName("全限定名"):用的最多
Class<?> clz3 = Class.forName("cn.sxt.reflect.User");
System.out.println(clz3);
}
}
3.获取九大内置类的字节码实例
对于对象来说,可以直接使用对象.getClass()或者Class.forName(className); 类名.class都可以获取Class实例.
但是我们的基本数据类型,就没有类的权限定名,也没有getClass方法.
问题: 那么如何使用Class类来表示基本数据类型的Class实例?
八大基本数据类型和 void关键字都是有 字节码实例的
byte,short,int,long,char,float,double,boolean ,void关键字
答 : 数据类型/void.class 即可
每个基本数据类型都是包装类型 如 :int ----Integer包装类型
注意: 基本数据类型和包装数据类型底层的字节码实例是不相同
//获取8大基本数据类型和void的字节码实例
//byte,short,int,long,char,float,double,boolean ,void关键字
public class BaiscDataTypeClassTest {
@Test
public void testName() throws Exception {
//1.获取byte的字节码实例
Class<?> byteClz = byte.class;
System.out.println(byteClz);
//....
//获取void关键字的字节码实例
Class<?> voidClz =void.class;
System.out.println(voidClz);
//所有的基本数据类型都有包装类型
//int---Integer
//int 和Integer 的字节码是绝对不相等的
Class<?> intClz1 = int.class;
Class<?> intClz2 = Integer.class;
System.out.println(intClz1);
System.out.println(intClz2);
System.out.println(intClz1 == intClz2);
}
}
4.获取数组类型的字节码实例
表示数组的Class实例:
String[] sArr1 = {"A","C"};
Class clz = String[].class;//此时clz表示就是一个String类型的一位数组类型
所有具有相同元素类型和维数的数组才共享同一份字节码(Class对象);
注意:和数组中的元素没有一点关系.
package cn.sxt.reflect._01.getclass;
import static org.junit.Assert.*;
import org.junit.Test;
//获取数组类型的字节码实例
public class ArrayClassTest {
@Test
public void testName() throws Exception {
//定义数组
int[] arr = {1,2,3};
Class<int[]> clz1 = (Class<int[]>) arr.getClass();
System.out.println(clz1);
Class<int[]> clz2= int[].class;
System.out.println(clz2);
int[] arr2 = {2,3,4};
Class<int[]> clz3 = (Class<int[]>) arr2.getClass();
System.out.println(clz3);
System.out.println(clz1 == clz2);//true
System.out.println(clz1 == clz3);//true
}
}
5.构造函数-Construstor
5.1获取构造函数
类的构函数有 有参数构造函数,无参构造函数,公共构造函数,非公共构造函数,根据不同的构造函数 Class提供了几种获取不同构造函数的方法。
//通过字节码实例获取构造器
public class ConstructorTest {
@Test
public void testName() throws Exception {
//1.获取Student的字节码实例
Class<?> stuClz = Student.class;
//2.获取所有的公共构造函数
Constructor<?>[] cts1 = stuClz.getConstructors();
for (Constructor<?> ct : cts1) {
System.out.println(ct);
}
System.out.println("----------------------");
//3.获取所有的构造函数包括私有的
Constructor<?>[] cts2 = stuClz.getDeclaredConstructors();
for (Constructor<?> ct : cts2) {
System.out.println(ct);
}
System.out.println("----------------------");
//4.获取指定的构造函数(clz.getConstructor(...))只能获取公共的构造函数
Constructor<?> ct1 = stuClz.getConstructor();
System.out.println(ct1);
Constructor<?> ct2 =stuClz.getConstructor(String.class);
System.out.println(ct2);
//4.获取指定的构造函数(clz.getDeclaredConstructor(...))获取的构造函数和权限没有关系
Constructor<?> ct3=stuClz.getDeclaredConstructor(String.class,int.class);
System.out.println(ct3);
}
}
5.2调用构造函数创建对象
Constructor<T>类:表示类中构造器的类型,Constructor的实例就是某一个类中的某一个构造器
常用方法:
public T newInstance(Object... initargs):如调用带参数的构造器,只能使用该方式. 参数:initargs:表示调用构造器的实际参数
返回:返回创建的实例,T表示Class所表示类的类型
如果:一个类中的构造器可以直接访问,同时没有参数.,那么可以直接使用Class类中的newInstance方法创建对象.
public Object newInstance():相当于new 类名();
调用私有的构造器:
//使用构造器创建对象
public class NewInstanceTest {
@Test
public void testName() throws Exception {
//1.获取Student的字节码实例
Class<?> clz = Class.forName("cn.sxt.reflect.Student");
//1.1如果类有无参数公共构造函数,直接可以使用类的字节码实例就创建对象
Student stu0 = (Student) clz.newInstance();
//2.获取一个参数的构造函数
Constructor<Student> ct1 = (Constructor<Student>) clz.getConstructor(String.class);
//2.1.创建对象
Student stu1 = ct1.newInstance("东方不败");
//3.获取私有构造函数并创建对象
Constructor<Student> ct2 = (Constructor<Student>) clz.getDeclaredConstructor(String.class,int.class);
//3.1设置权限可以创建对象
ct2.setAccessible(true);
//3.2创建对象
Student stu2 = ct2.newInstance("西门吹雪",50);
}
}
6.操作方法-Method
一个类创建对象以后,一般就要执行对象的方法等等,使用反射操作对象
首先要获取方法,再去执行.。
package cn.sxt.reflect._03method;
import java.util.Arrays;
public class Person {
public void hell1() {
System.out.println("我是无参数无返回值方法");
}
public String hello2(String name) {
return "你好 :"+name;
}
private String hello3(String name,int age) {
return "我是 :"+name+",今年 :"+age;
}
public static void staticMethod(String name) {
System.out.println("我是静态方法 :"+name);
}
public static void method1(int[] intArr) {
System.out.println(Arrays.toString(intArr));
}
public static void method2(String[] strArr) {
System.out.println(Arrays.toString(strArr));
}
}
package cn.sxt.reflect._03method;
import static org.junit.Assert.*;
import java.lang.reflect.Method;
import org.junit.Test;
//获取Person类的方法
public class GetMethodTest {
@Test
public void testName() throws Exception {
// 1.获取Person字节码实例
Class<Person> clz = Person.class;
// 2.创建对象
Person p = clz.newInstance();
// 3.获取方法(使用反射),获取所有公共方法,包含父类的公共方法
Method[] methods1 = clz.getMethods();
for (Method method : methods1) {
System.out.println(method);
}
System.out.println("------------------------------");
// 4.获取自己类中的所有方法(包括私有)
Method[] methods2 = clz.getDeclaredMethods();
for (Method method : methods2) {
System.out.println(method);
}
System.out.println("------------------------------");
// 4.获取单个指定名称的方法
Method method = clz.getMethod("hello2", String.class);
System.out.println(method);
// 4.1执行方法
Object res = method.invoke(p, "陆小凤");
System.out.println(res);
// 5.获取私有的方法
Method hello3 = clz.getDeclaredMethod("hello3", String.class, int.class);
System.out.println(hello3);
// 5.1设置忽略访问权限
hello3.setAccessible(true);
Object res1 = hello3.invoke(p, "叶孤城", 30);
System.out.println(res1);
// 6.获取静态方法
Method staticMethod = clz.getMethod("staticMethod", String.class);
// 6.1执行静态方法
staticMethod.invoke(null, "花满楼");
// 7.获取有整数数组参数的方法
Method method1 = clz.getMethod("method1", int[].class);
method1.invoke(null, new Object[] {new int[] { 1, 2, 3, 4 }});
// 8.获取有整数数组参数的方法
/*
* 如果反射传递的参数是引用类型,底层有一个拆箱的功能,会将数组的元素拆成一个个参数传递过来
* 解决方案: 将数组外面在包装一层数组,如果拆箱一次,得到还是一个数组
*/
Method method2 = clz.getMethod("method2", String[].class);
method2.invoke(null,new Object[] {new String[] {"AA","BB","CC"}});
}
}
package cn.sxt.reflect._03method;
import java.util.Arrays;
public class Person {
public static void method1(int... intArr) {
System.out.println(Arrays.toString(intArr));
}
public static void method2(String...strArr) {
System.out.println(Arrays.toString(strArr));
}
}
// 7.获取有整数数组参数的方法
Method method1 = clz.getMethod("method1", int[].class);
method1.invoke(null, new Object[] {new int[] { 1, 2, 3, 4 }});
// 8.获取有整数数组参数的方法
/*
* 如果反射传递的参数是引用类型,底层有一个拆箱的功能,会将数组的元素拆成一个个参数传递过来
* 解决方案: 将数组外面在包装一层数组,如果拆箱一次,得到还是一个数组
*/
Method method2 = clz.getMethod("method2", String[].class);
method2.invoke(null,new Object[] {new String[] {"AA","BB","CC"}});
7.操作字段(成员变量)--Field
类中的字段有各种数据类型和各种访问权限,针对这些情况,反射操作有对应的方法来获取和处理
package cn.sxt.reflect._04Field;
import static org.junit.Assert.*;
import java.lang.reflect.Field;
import org.junit.Test;
public class FieldTest {
@Test
public void testName() throws Exception {
//1.获取People字节码
Class<People> clz = People.class;
People p = clz.newInstance();
//2.获取所有公共字段
Field[] fields1 = clz.getFields();
for (Field field : fields1) {
System.out.println(field);
}
System.out.println("---------------------");
//3.获取所有字段,和访问权限无关
Field[] fields2 = clz.getDeclaredFields();
for (Field field : fields2) {
System.out.println(field);
}
System.out.println("---------------------");
//3.获取指定的公共字段
Field emialField = clz.getField("emial");
System.out.println(emialField);
//为字段设置值
emialField.set(p, "zhagnsan@qq.com");
System.out.println(p);
//4.获取指定所有的字段,和访问权限无关
Field nameFiled = clz.getDeclaredField("name");
System.out.println(nameFiled);
//设置忽略访问权限
nameFiled.setAccessible(true);
nameFiled.set(p, "张三");
System.out.println(p);
//5 获取age字段
Field ageFile = clz.getDeclaredField("age");
ageFile.setAccessible(true);
//设置忽略访问权限
ageFile.setInt(p, 18);
System.out.println(p);
}
}
8.Class的其他常用API方法
//Class字节码的其他api
@Test
public void testName() throws Exception {
//1.获取UserDaoImpl的字节码实例
Class<?> clz = Class.forName("cn.sxt.reflect._05otherapi.UserDaoImpl");
//2.获取所有的接口
Class<?>[] interfaces = clz.getInterfaces();
for (Class<?> intface : interfaces) {
System.out.println(intface);
}
//3.获取全限定名
System.out.println(clz.getName());
//4.获取简单类名
System.out.println(clz.getSimpleName());
//5.获取包
System.out.println(clz.getPackage().getName());
}