zoukankan      html  css  js  c++  java
  • 关于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"));

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


  • 相关阅读:
    期中考试补完计划
    c++简单程序设计-4
    c++简单程序设计-3
    c++简单程序设计-2
    c++简单程序设计-1
    layer ifram 弹出框
    定时发邮件
    二维码生成
    文件或文件夹的压缩
    excel 导入 与 导出
  • 原文地址:https://www.cnblogs.com/lkzf/p/3943337.html
Copyright © 2011-2022 走看看