Fork me on GitHub

java反射

一、Class类的使用

类是对象,任何一个类都是java.lang.class类的实例对象,这个类,这个实例对象可以有三种表达方式,比如Student类:

  • 任何一个类都有一个隐含的静态成员变量class
1
Class c1 = Student.class;
  • 通过getClass方法获得
1
Class c2 = Student.getClass();
  • forName
1
Class c3 = Class.forName("com.Student");

c1,c2,c3表示了Student类的类类型(class type),可以通过类的类类型创建该类的对象实例—->通过c1 or c2 or c3创建Student的实例对象

1
Student stu = c1.newInstance();


二、动态加载类

  • 静态加载类:编译时加载的类,通过new创建对象是静态加载类
  • 动态加载类:运行时加载的类,编译时不用管类是否存在或者是否错误等问题

三、获取方法信息和成员变量,构造函数信息

方法信息

  • 所有的public的函数,包括父类继承而来的
1
Method[] ms = c.getMethods();
  • 所有该类自己声明的方法,不问访问权限
1
Method[] ms = c.getDeclaredMethods();
  • 返回值类型的类类型
1
Class returnType = ms[i].getReturnType();
  • 参数列表的类型的类类型
1
Class[] paramTypes = ms[i].getParameterTypes();
  • 方法的名称
1
String methodName = ms[i].getName();

成员变量

  • 所有的public的成员变量的信息
1
Field[] fs = c.getFields();
  • 该类自己声明的成员变量的信息
1
Field[] fs = c.getDeclaredFields();
  • 成员变量的类型的类类型
1
Class fieldType = field.getType();
  • 成员变量的名称
1
String fieldName = field.getName();
  • 构造函数
  • 所有的public构造函数
1
Constructor[] cs = c.getConstructors();
  • 所有的构造函数
1
Constructor[] cs = c.getDeclaredConstructors();
  • 构造函数的参数列表,得到的是参数列表的类类型
1
Class fieldType = field.getType();
  • 获取类的私有属性或私有方法
    直接获取私有属性或者方法会报java.lang.IllegalAccessException异常,AccessibleObject是Class、Method、Field的父类,通过设置setAccessible(true)可以获取类的私有属性,获取和调用私有方法

Student类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public class Student {

private String name;

private int age;

public Student(String name, int age) {
this.name = name;
this.age = age;
}

private void speck(String name) {
System.out.println("I am " + name);
}

@Override
public String toString() {
return "name: " + name + "age: " + age;
}
}

获取私有属性:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Student student = new Student("pinnuli", 18);
Class studentClass = student.getClass();
Field[] studentField = studentClass.getDeclaredFields();

for (int i = 0; i < studentField.length; i++) {
studentField[i].setAccessible(true);
try {
System.out.println(studentField[i].get(student));
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
System.out.println(studentField[i].getName());
}

获取私有方法

1
2
3
4
5
6
7
Student student = new Student("pinnuli", 18);
Class studentClass = student.getClass();
Method[] studentMethods = studentClass.getDeclaredMethods();
for (int i = 0; i < studentMethods.length; i++) {
studentMethods[i].setAccessible(true);
System.out.println(studentMethods[i].getName());
}

调用私有方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
Student student = new Student("pinnuli", 18);
Class studentClass = student.getClass();
Method[] studentMethods = studentClass.getDeclaredMethods();
for (int i = 0; i < studentMethods.length; i++) {
studentMethods[i].setAccessible(true);
try {
studentMethods[i].invoke(student,"pinnuli");
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
System.out.println(studentMethods[i].getName());
}

更多方法看API文档


四、方法反射的基本操作

现有A类如下:

1
2
3
4
5
6
7
8
9
10
11
class A{
public void print(){
System.out.println("helloworld");
}
public void print(int a,int b){
System.out.println(a+b);
}
public void print(String a,String b){
System.out.println(a.toUpperCase()+","+b.toLowerCase());
}
}

1.要获取一个方法就是获取类的信息,获取类的信息首先要获取类的类类型

1
2
A a1 = new A();
Class c = a1.getClass();

2.通过名称和参数列表来获取方法

1
Method m = c.getMethod("print", int.class,int.class);

3.方法的反射操作

1
Object o = m.invoke(a1, 10,20);

这个操作的效果等同于a1.print(10,20)

通过方法的反射,我们可以绕过编译

参阅:
慕课网:反射——Java高级开发必须懂的

-------------本文结束感谢您的阅读-------------