1、反射的基础
  反射的基石----》Class类
  java程序中的各个java类属于同一类事物,描述这类事物的java类名就是Class

  获取字节码对应的实例对象(Class类型)
  class cls1 = Person.class//字节码1;
  p1.getClass();
  Class.forName("java.lang.String"); 主要用这种进行反射

  8个基本类型,对应着8个Class对象
  void.class 对象

String str1 = "abc";
    Class cls1 = str1.getClass();
    Class cls2 = String.class;
    Class cls3 = Class.forName("java.lang.String");
    cls1==cls2 //true
    cls1==cls3 //true
    int.class.isPrimitive //true
    int.class == Integer.class //flase
    int.class == Integer.TYPE //true
    数组类型的Class实例对象 Class.isArray()

2反射的概念
  反射就是把java类中的各种成分映射成对应的java类
  例如:
  成员变量---》Field对象
  方法----》Method对象
  构造方法---》Constructor对象
  ...

3、 构造方法的反射应用

  Constructor类代表某一个构造方法

//new String(new StringBuffer("abc"))
  Constructor constructor1 = String.class.getConstructor(StringBuffer.class);
  String str2=(String)constructor1.newInstance(new StringBuffer("abc"))

  class -> constructor -> new object

  反射会导致程序性能下降

  Class.newInstance()方法
  例如:String obj = (Class)Class.forName("java.lang.String").newInstance();
  该方法内部先得到默认的构造方法,然后用该构造方法创建实例对象
  用了缓存机制来保存默认构造方法的实例对象


4、成员变量的反射

Field类代表某各类中的一个成员变量

    public class ReflectPoint{
        private int x;
        public int y;
        ...构造方法 
    }

    ReflectPoint pt1 = new ReflectPoint(3,5);
    //fieldY不是对象身上的变量,而是类上的,要用它取对应对象上的属性值
    Field fieldY = pt1.getClass().getField("y"); //public
    fieldY.get(pt1); // 5 
    Field fieldX = pt1.getClass().getDeclaredField("x");//private
    filedX.setAccessible(true);//暴力反射
    fieldX.get(pt1);


    public class ReflectPoint{
        public String str1 ="ball";
        public String str2 ="basketball";
        public String str3 ="itcast";
    }

    private static void changeStringValue(Object obj){
        Field[] fields = obj.getClass.getFields();
        for(Field field:fields){
            //对字节码的比较用==号比 因为同一份字节码
            if(field.getType() == String.class){
                String oldValue =(String)field.get(obj);
                String newValue = oldValue.replace('b','a');
                field.set(obj,newValue);
            }
        }
    }

 

5 、成员方法的反射

  Method类代表某个类中的一个成员方法

Method methodCharAt = String.class.getMethod("charAt",int.class);
  //调用方法一定是在一个对象上
  String str1="abc";
  methodCharAt.invoke(str1,1);

  如果传递给Method对象的invoke()方法的第一个参数为null,说明
  该Method对象对应的是一个静态方法

 

6 、对接受数组参数的成员方法进行反射

  用反射方式执行某个类中的main方法

public class TestArguments{
        public static void main(String[] args){
            for(String arg:args){
                System.out.println(arg);
            }
        }
    }

    String clazz = "com.test.TestArguments";
    Method mainMethod = Class.forName(clazz).getMethod("main",String[].class);
    mainMethod.invoke(null,new Object[](new String[]{"111","222"}));
    mainMethod.invoke(null,(Object)new String[]{"111","222"});
    //编译器会做特殊处理,编译时不将参数作为数组看待,也就不会讲数组打算成若干参数了

 

7、 数组与Object的反射

  相同的元素且相同的维度则是同一份字节码文件

int[] a1 = new int[3];
    int[] a2 = new int[4];
    int[][] a3 = new int[2][3];
    String[] a4 = new String[3];
    a1.getClass() == a2.getClass();//true
    a1.getClass() == a3.getClass();//false
    a1.getClass() == a4.getClass();//false

基本类型的数组是不能转为Object数组
Object[] aobj = a1//error

具有相同维数和元素类型的数组属于同一个类型,即具有相同的Class实例对象

代表数组的Class实例对象的getSuperclass()方法返回的父类为Object类对应的Class

基本类型的一维数组可以被当做Object类型使用,不能当作Object[]类型使用,非基本类型的一维数组,既可以当做Object类型使用,又可以当做Object[]类型使用

Arrays.asList()方法处理int[]和String[]存在差异Arrays工具类用于完成对数组的反射操作

private static void printObject(Object obj){
        Class clazz = obj.getClass();
        if(clazz.isArray()){
            int len = Array.getLength();
            for(int i=0;i<=len;i++){
                System.out.println(Array.get(obj,i));
            }
        }else{
            System.out.println(obj);
        }
    }