反射
反射
java的反射机制大致是这样的:通过某个类的class文件得到整个类的完整结构信息
我们说,万事万物都是对象,因此对于一个类来说,我们也可以将类中的一些属性看成是对象。整个类可以看成是Class对象,构造函数可以看做是Construtor对象,字段可以看成是Field对象,一般方法可以看成是Method方法。
其实反射机制,通俗的说就是我们可以通过class将与这个类有关的对象全部找出来
那么我们要先学会如何通过Class字节码文件加载该类对象呢?
其实通过Class加载类的方法有三个:比如说我们要加载的类是User
Class c1=User.class;
User user=new User();
Class c2=user.getClass();
Class c3=Class.forName("xx.xx.User");
其中第三中加载方法中需要注意的是,右边括号中所要填写的是该类的全类名称,就是路径,也就是哪个包下面的哪个类
反射常见方法介绍:
- getFields()与getDeclareFields()
相同点:两者都可以获取类中的字段
不同点: getFields():拿到的是public修饰符修饰的字段 getDeclareFields()拿到的是类中的所有字段需要注意的是:当要使用拿到的private修饰符修饰的字段时,由于Java会进行访问检查,所以必须设置setAccessible(true)
- getMethods()和getDeclareMethods()
相同点:两者都获取类中的成员方法
不同点:getMethods()只获取类中的公共成员方法,若该类存在父类,则会获得从父类继承过来的方法;而getDeclareMethods()只是获取类中的成员方法
- Field常用方法:
1、Field.getName():获取字段的名称
2、Field.get(Object obj):获取字段的值
3、Field.set(Object obj, Object value):为字段赋值
- Method常用方法:
1、Method.getName():获取方法的名称
2、Method.getReturnType():获取方法的返回值类型
3、Method.getParameterCount():获取方法的参数个数
4、Method.invoke(Object obj, Object… args):通过obj对象,执行该方法
- Class常用方法:(上面介绍过的 不再介绍)
1、getConstructors():获取类中的所有构造函数
2、getConstructor(Class… parameterTypes):获取指定参数的构造函数,可用于生成对象实例
3、newInstance():创建该class的对象实例
4、getName():获取类的全名 如com.User
5、getSimpleName():获取类的简称 如User
6、getField(String name):获取某指定名称的字段
7、getMethod(String name, Class… parameterTypes):获取指定名称、参数的方法
- 其他常用方法
getModifiers():以int方式返回调用者的修饰符 常用对照关系:public:1 private:2
代码举例:
public class User implements Serializable {
private static final long serialVersionUID = -3015143840907090098L;
private String name;
private String age;
public User(String name, String age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getAge() {
return age;
}
public void setAge(String age) {
this.age = age;
}
}
以User类为例,进行展示:
Class<User> clazz = User.class;
String name = clazz.getName();
String simpleName = clazz.getSimpleName();
System.out.println("name:" + name);
System.out.println("simpleName:" + simpleName);
运行结果如下:
name:com.wojiushiwo.User
simpleName:User
//获得指定参数类型的构造函数,并使之实例化
Constructor<User> constructor = clazz.getConstructor(String.class, String.class);
User newInstance = constructor.newInstance("Hello", "23");
//获取类中的字段
Field[] fields = clazz.getDeclaredFields();
for (Field field : fields) {
System.out.println(field.getName());
}
运行结果:
serialVersionUID
name
age
//当要读写字段时,由于是private修饰,所以要setAccessible(true)
Field[] fields = clazz.getDeclaredFields();
for (Field field : fields) {
field.setAccessible(true);
if ("serialVersionUID".equals(field.getName())) {
System.out.println(field.getName());
//取serialVersionUID的值
long value = field.getLong(newInstance);
System.out.println("serialVersionUID:" + value);
} else {
// 普通属性的值
Object object = field.get(newInstance);
System.out.println("属性的值:" + object);
}
}
运行结果:
serialVersionUID:-3015143840907090098
属性的值:Hello
属性的值:23
Method[] methods = clazz.getMethods();
for (Method method : methods) {
System.out.print(method.getName()+" ");
System.out.print(method.getReturnType()+" ");
System.out.print("参数个数:" + method.getParameterCount());
System.out.println();
}
运行结果:
getName class java.lang.String 参数个数:0
setName void 参数个数:1
setAge void 参数个数:1
getAge class java.lang.String 参数个数:0
wait void 参数个数:0
wait void 参数个数:2
wait void 参数个数:1
equals boolean 参数个数:1
toString class java.lang.String 参数个数:0
hashCode int 参数个数:0
getClass class java.lang.Class 参数个数:0
notify void 参数个数:0
notifyAll void 参数个数:0若将方法改为getDeclareMethods(),则运行结果如下:
getName class java.lang.String 参数个数:0
setName void 参数个数:1
getAge class java.lang.String 参数个数:0
setAge void 参数个数:1
以上是个人觉得常用方法。如有错误,请纠正!