反射

JAVA反射机制是在运行状态中,对于任意一个类(class文件),都能够知道这个类的所有属性和方法;

对于任意一个对象,都能够调用它的任意一个方法和属性。

这种动态获取的信息以及动态调用对象的方法的功能称为JAVA语言的反射机制。

动态获取类中信息,就是JAVA反射。可以理解为对类的解剖。

设有以下类:

public class Person
{
     private int age;
     private String name;
     public Person(int age,String name){
         super();
         this.page=age;
         this.name=name;
         System.out.println("Person param run...");
    }
    public Person(){
         super();
         System.out.println("person run");
    }
    public void show(){
        System.out.priontln(name+"...show run ..."+age);
    }
    private void method(){
       System.out.println("method run ");
   }
    public void paramMethod(String str,int num){
        System.out.println("paramMethod run..."+str+":"+num);
   }
   public static void staticMethod(){
      System.out.println("static method run...");
  }
}

要想对字节码文件进行解剖,必须要用字节码文件对象。

如何获取字节码文件对象呢?

/*
*获取字节码对象的方式:
*1.Object类中的getClass()方法的。
*想要用这种方法,必须要明确具体的类,并创建对象
*比较麻烦。
*/
public static void getClassObjectd_1(){
     Person p=new Person();
     Class clazz =p.getClass();
     Person p1=new Person();
     Class clazz1=p1.getClass();
     System.out.println(clazz==clazz1);
}

/*
*2.任何数据类型都具备一个静态的属性.class来获取其对应的Class对象。
*相对简单,但是还是需要明确用到类中的静态成员,也需要知道类名,还是不够扩展
*/
public static void getCalssObject_2(){
    Class clazz=Person.class;
    Class clazz1=Person.calss;
    System.out.println(clazz==clazz1);
}

/*
*3.只要通过给定的类的字符串名就可以获取该类,更为扩展。
*可以用Class类中的方法完成。
*该方法就是forName()方法
*这种方式只要有名称即可,更为方便,扩展性更强,尽量用此方法。
*/
public static void getCalssObject_3() throws ClassNotFoundException{
    String className="cn.sabc.com.Person"; //此处类名必须包含包名
    Class clazz=Class.forName(className);
    System.out.print(clazz);
}

 用new类方法和forName()有什么区别?

public static void CreateNewObject(){
    //早期:new的时候,先根据被new的类的名称找寻该类的字节码文件,并加载进内存,并创建该字节码对象,并接着创建该字节文件的对应的Person对象。
   cn.sabc.com.Person p=new cn.sabc.com.Person();
   //现在:
   String name="cn.sabc.com.Person";
   //找寻该名称类文件,并加载进内存,并产生Class对象。
   Class clazz=Class.forName(name);
   //如何产生该类的对象呢?
   Object obj=clazz.newInstance();
}


public static void CreateNewObject_1(){
    //当获取指定名称对应类中的所体现的对象时,而该对象初始化不命名用空参构造函数怎么办呢?
   cn.sabc.com.Person p=new cn.sabc.com.Person(39,"小强”);
   //既然是通过指定的构造随叫随到数进行对象的初始化,所以应该先获取到该构造函数。
   //通过字节码码文件对象即可完成。利用getConStructor(paramterTypes),可获取所有公有的构造函数
   String name="cn.sabc.com.Person";
   Class clazz=Class.forName(name);
  //获取到了指定的构造函数对象
   Constructor constuctor=clazz.getConstructor(int.class,String.class);
  //通过该构造器对象的newInstance方法进行对象的初始化
 object obj=constructor.newInstance(38,"小明");
}

 获取一个类对象中的字段。

public static void getFieldDemo(){
    Class clazz=Class.forName("cn.sabc.com.Person");
   // Field field=clazz.getField("age");  //不能获取私有字段
    Field field1=clazz.getDeclareField("age");//能获取本类,但包含私有
   System.out.println(field1);
   //对私有字段的访问取消权限检查,暴力访问,不建议使用
   field.setAccessible(true);
   Object obj=clazz.newInstance(); 
   field.set(89,obj);
   Object o=field.get(obj); //
   System.out.println(o);
}

获取一个类对象的方法

public static void getMethodDemo(){
    Class clazz=Class.forName("cn.sabc.com.Person");
   // Method[] methods=clazz.getMethods(); //只获取公有的
    Method[] methods=clazz.getDeclareMethods(); //只获取本类中所有的方法,包含私有
    for(Method method :methods)
   {
        System.out.println(method);

    }
}

//执行无参方法
public static void getMethodDemo_2(){
    Class clazz=Class.forName("cn.sabc.com.Person");
    Method method=clazz.getMethod("show",null);
    //调用方法需要有对象
   // Object obj=clazz.newInstance(); 无参构造函数 
    Constuctor constructor=clazz.getConstructor(String.class,int.class);
    Objet obj=constructor.newInstance(37,"小明");
    method.invoke(obj,null);  //执行该方法
    }
}

//执行有参方法
public static void getMethodDemo_3(){
    Class clazz=Class.forName("cn.sabc.com.Person");
    Method method=clazz.getMethod("paramMethod",int.class,String.class);
    //调用方法需要有对象
   // Object obj=clazz.newInstance(); 无参构造函数 
    method.invoke(obj,37,"小强");  //执行该方法
    }
}

 

posted @ 2017-10-30 16:11  jsddj  阅读(117)  评论(0编辑  收藏  举报