知识点1:获取类字节码的三种形式
1、Class date = Date.class;//根据类名获取字节码
2、Date date= new Date();
date.getClass();//对象的形式获取字节码
3、Class.forName("java.util.Date");//完整名字获取字节码
知识点2:反射就是把java类中的各种成分映射成相应的java类。
知识点3:构造方法的反射:Constructor类代表某个类中的一个构造方法。
//得到某个类所有的构造方法
Constructor[] constructors = Class.forName("java.lang.String").getConstructors(); //得到某一个构造方法
Constructor constructor = Class.forName("java.lang.String").getConstructor(StringBuffer.class);
通常方式:new String(new StringBuffer("abc"));
反射方式:利用Constructor类实现
//第一步:创建String类的构造方法,参数为StringBuffer。
Constructor constructor1 = String.class.getConstructor(StringBuffer.class)
//第二步:根据单例创建对象
String str = (String)constructor1.newInstance(new StringBuffer("abc"));
注意:StringBuffer.class为了创建构造方法,new StringBuffer("abc")为了创建对象,二者类名必须一致。 //Class.newInstance()方法
例子:String obj = Class.forName("java.lang.String").newInstance();
该方法先得到默认的构造方法,再用该构造方法创建实例对象
该方法内部用到了缓存机制保存默认构造方法的实例对象。
知识点4:成员变量的反射:Field类代表某个类中的一个成员变量。
问题:得到的Field对象是对应到类上面的成员变量,还是对应对象上的成员变量?答案是对应类上面的成员变量。
ReflectPoint pt1 = new ReflectPoint(3,5);//ReflectPoint有两个成员变量,private x和public y
Field fieldy = pt1.getClass().getField("y");//根据对象名获取字节码,通过getField()方法传入成员变量名获取成员变量
int y = fieldy.get(pt1);//根据指定对象获取成员变量的值 Field fieldx = pt1.getClass().getDeclaredField("x");//对于私有的成员变量,通过getDeclaredField()方法获取成员变量
fieldX.setAccessible(true);//由于是私有的,需要通过该方法强行获取值
int x= fieldx.get(pt1);//根据指定对象获取成员变量的值
例子:利用反射修改成员变量的值
private static void changeStringValue(Object obj) throws Exception{
Field[] fields = obj.getClass().getField();//根据传过来的形参对象获取所有成员变量
for(Field field:fields){
if(field.getType() == String.class){//判断类型是否等于String
String oldValue = (String)field.get(obj);
String newValue = olldValue.replace('b', 'a');
field.set(obj, newValue);//设置成员变量为新的值
}
}
}
知识点5:方法的反射:Method类代表某个类中的一个成员方法。
//得到一个类中的某一个方法,需要传入方法的名字和参数。
Method charAt = String.class.getMethod("charAt",int.class); 调用形式
str.charAt(1);//通常方式
charAt.invoke(str,1);//反射方式,参数为对象名称和参数。
注意:如果传递给Method对象的invoke()方法的一个参数为null,说明该Method对象对应的是一个静态方法。 main方法的反射:
Class class = Class.forName(arg[0]);//args[0]代表类名
Method methodMain = class.getMethod("main",String[].class);
methodMain.invoke(null,new Object[]{new String[]{"aaa","bbb"}});//因为main方法是静态方法,所以第一个参数为null
methodMain.invoke(null,(Object)new String[]{"aaa","bbb"});//上面这两种方式都可以
知识点6:数组的反射
1、具有相同维数和元素类型的数组属于同一个类型,即具有相同的Class实例对象
2、代表数组的Class实例对象的getSuperclass()方法返回的父类为Object类对应的Class。
3、基本类型的一维数组可以被当做Object类型使用,不能当做Object[]类型使用;非基本类型的一维数组,既可以当做Object类型使用,也可以当做Object[]类型使用。
4、Arrays.asList()方法处理int[]和String[]时有差异。int[]使用输出[I@hash值,String可以输出具体值:a,b,c
5、Arrays工具类用于完成对数组的反射操作。