关于java reflect

反射的基石 Class类

  • 对比提问: Person类代表人,它的实例对象就是张三,李四这样一个个具体的人, Java程序中的各个Java类属于同一类事物,描述这类事物的Java类名就是Class。对比提问:众多的人用一个什么类表示?众多的Java类用一个什么类表示?
    • 人 Person
    • Java类 Class
  • Class类代表Java类,它的各个实例对象又分别对应什么呢?
    • 对应各个类在内存中的字节码,例如,Person类的字节码,ArrayList类的字节码,等等。
    • 一个类被类加载器加载到内存中,占用一片存储空间,这个空间里面的内容就是类的字节码,不同的类的字节码是不同的,所以它们在内存中的内容是不同的,这一个个的空间可分别用一个个的对象来表示,这些对象显然具有相同的类型,这个类型是什么呢?
  •  何得到各个字节码对应的实例对象( Class类型)
    • 类名.class,例如,System.class
    • 对象.getClass(),例如,new Date().getClass()
    • static Class.forName("类名"),例如,Class.forName("java.util.Date");
      • Class.forName()得到字节码的情况:
           1、字节码已经加载到java虚拟机中,去得到字节码
           2、java虚拟机中还没有生成字节码 用类加载器进行加载,加载的字节码缓冲到虚拟机中
  • 九个预定义Class实例对象:八个基本数据类型和void类型
    • 参看Class.isPrimitive方法的帮助
    • Int.class == Integer.TYPE]
    1. byte.class
    2. char.class
    3. short.class
    4. int.class
    5. long.class
    6. float.class 
    7. double.class
    8. boolean.class和
    9. void.class)
  • 数组类型的Class实例对象
    • Class.isArray()
  • 总之,只要是在源程序中出现的类型,都有各自的Class实例对象,例如,int[],void…

1.Java类用于描述一类事物的共性,该类事物有什么属性,没有什么属性,至于这个属性的值是什么,则是由这个类的实例对象来确定的,不同的实例对象有不同的属性值。Java程序中的各个Java类,它们是否属于同一类事物,是不是可以用一个类来描述这类事物呢?

这个类的名字就是Class,要注意与小写class关键字的区别哦。Class类描述了哪些方面的信息呢?

类的名字,类的访问属性,类所属于的包名,字段名称的列表、方法名称的列表,等等。

学习反射,首先就要明白Class这个类。写如下代码进行对比理解:

        /*Person p1 = new Person("zhangsan");
        Person p2 = new Person("lisi");
        */
        /*Class x1 = Vector类在内存里的字节码
         Class x2 = Date类在内存里的字节码*/
        Class x1 = Vector.class;
        Class x2 = Date.class;

 每个java类都是Class的一个实例对象,它们的内容不同,但是,它们的特征相同,譬如,都有方法,有字段,有父类,有包。

2. ******讲课时要一定画一张图:多个类的字节码装载入内存,在内存中加入一个个方块空间表示字节码,然后用一个个椭圆表示以这个字节码创建出来的实例对象,并用监视代码来说明字节码只被装载一次,而它构造的实例对象的构造方法被调用了多次。

用如下代码更进一步说明Class的实例是什么?是一份字节码,一个类在虚拟机中通常只有一份字节码:

  

Date d1 = new Date();

Class clazz1 = d1.getClass();

Class clazz2 = Date.class;

Class clazz3 = null;

clazz3 = Class.forName("java.util.Date");

 

  if(clazz1==clazz2)

  {

      System.out.println(clazz1.getName());

  }

  if(clazz1==clazz3)

  {

      System.out.println(clazz1.getName());

  } 

3.一个奇怪的问题:加载了字节码,并调用了其getMethods之类的方法,但是没有看到类的静态代码块被执行,只有在第一个实例对象被创建时,这个静态代码才会被执行。

准确的说,静态代码块不是在类加载时被调用的,而是第一个实例对象被创建时才执行的。


  • 反射就是把Java类中的各种成分映射成相应的java类。
      • 例如,一个Java类中用一个Class类的对象来表示,一个类中的组成部分:成员变量,方法,构造方法,包等等信息也用一个个的Java类来表示,就像汽车是一个类,汽车中的发动机,变速箱等等也是一个个的类。表示java类的Class类显然要提供一系列的方法,来获得其中的变量,方法,构造方法,修饰符,包等信息,这些信息就是用相应类的实例对象来表示,它们是Field、Method、Contructor、Package等等。
  • 一个类中的每个成员都可以用相应的反射API类的一个实例对象来表示,
      • 通过调用Class类的方法可以得到这些实例对象后,得到这些实例对象后有什么用呢?怎么用呢?这正是学习和应用反射的要点。

Constructor类

  • Constructor类代表某个类中的一个构造方法
  • 得到某个类所有的构造方法:
    •   例子:
      Constructor [] constructors= Class.forName("java.lang.String").getConstructors();
  • 得到某一个构造方法:
    •   例子:      
       Constructor constructor = Class.forName(“java.lang.String”).getConstructor(StringBuffer.class);
  •   //获得方法时要用到类型
  • 创建实例对象:
    • 通常方式:
      String str = new String(new StringBuffer("abc"));
    • 反射方式: 
      String str = (String)constructor.newInstance(new StringBuffer("abc"));
    • //调用获得的方法时要用到上面相同类型的实例对象
  • Class.newInstance()方法:
    • 例子:String obj = (String)Class.forName("java.lang.String").newInstance();
    • 该方法内部先得到默认的构造方法,然后用该构造方法创建实例对象。
    • 该方法内部的具体代码是怎样写的呢?用到了缓存机制来保存默认构造方法的实例对象。

2. 一个类有多个构造方法,用什么方式可以区分清楚想得到其中的哪个方法呢?根据参数的个数和类型,例如,Class.getMethod(name,Class... args)中的args参数就代表所要获取的那个方法的各个参数的类型的列表。重点:参数类型用什么方式表示?用Class实例对象。例如:

int.class,(int []).class

int [] ints = new int[0];

ints.getClass();

Constructor对象代表一个构造方法,大家觉得Constructor对象上会有什么方法呢?得到名字,得到所属于的类,产生实例对象。

3. 讲解创建实例对象时,先应该举例说说通常情况下是怎样做的, String str = new String(new StringBuffer(“abc”));在源程序中用注释方式给出传统方式的代码,以便对比提醒程序阅读者。

然后再说用反射如何做

      String str = (String)constructor.newInstance(/*"abc"*/new StringBuffer("abc"));

      System.out.println(str);

讲反射方式创建实例对象时,先故意用string作为参数传进去,根据错误让大家感受到确实是那个构造方法,然后再改为传一个StringBuffer类型的参数进去,  String str = (String)constructor.newInstance(/*"abc"*/new StringBuffer("abc"));

好比,我叫来一个吃人不吃草的恐龙,等到它要吃东西时,我得给他送真人去了吧。


原文地址:https://www.cnblogs.com/lkzf/p/3943337.html