首页 > 人工智能 > 正文

Java开发中类的加载及反射机制

2009-03-30 08:53:25  来源:CIO 时代网

摘要:JAVA中类文件加载是动态的。JVM指令是被封装在了。class文件里面,而。class文件的加载过程是动态的,也就是说当我们用到的时候才会去加载,如果不用的话,就不会去加载我们的类。
关键词: Java 开发中

 JAVA 中类文件加载是动态的。JVM指令是被封装在了。class文件里面,而。class文件的加载过程是动态的,也就是说当我们用到的时候才会去加载,如果不用的话,就不会去加载我们的类。这里所说的用到包括两种方式,第一种就是new一个对象的时候(这个时候要特别注意,当设计到多态的时候,就会有一点点变化,这时候编译器会做一些优化,这样以来当加载的时候会提前加载设计到多态的类,关于这一点下面有个例子(example 1)来说明。另一种就是当一个类的静态代码被调用的时候。
    java 代码
    //example 1
    //Zoo.java
    abstract class Animal {
    Animal(){
    System.out.println(“Animal constructor”);
    }
    }
    class Tiger extends Animal {
    Tiger(){
    System.out.println(“Tig constructor ”);
    }
    }
    class Dog extends Animal {
    Dog(){
    System.out.println(“Dog Constructor ”);
    }
    }
    public class Zoo {
    private Animal am; //Example 1.1
    //private Dog am; Example 1.2
    private Tiger tiger;
    Zoo(){
    tiger = new Tiger();
    am = new Dog();
    }
    public static void main(String [] args){
    System.out.println(“new Zoo before”);
    Zoo z = new Zoo();
    System.out.println(“new Zoo after ”);
    }
    }
    我们可以看出:当我们将子类对象赋值给父类时,编译器会做一点优化,于是加载器在还没有new子类对象的时候已经加载了父类以及子类(example1.1结果),当不存在多态的时候,我们可以看到是当要new Dog()的时候才会加载Dog以及父类。无论何种方式,在new之前,类确实已经加载到了内存中。
    JAVA为我们提供了两种动态机制。第一种是隐式机制。其实new一个对象和调用类的静态方法时,就是隐式机制在工作。第二种是显示机制。显示的机制又有两种策略(第一种是用java.lang.Class的forName(String str)方法,第二种是用java.lang.ClassLoader的loadClass())。
    第一种:利用forName方法
    当我们查API文档就会发现forName方法有两种形式。分别如下:
    public static Class<?> forName(String className)
    throws ClassNotFoundException
    public static Class<?> forName(String name,
    boolean initialize,
    ClassLoader loader)
    throws ClassNotFoundException
    先来说说第二种方法:第二个方法值得注意的就是第二个参数boolean initialize,如果我们把这个参数设置为false,那么当我们加载完类后就不会执行静态代码和静态的初始化动作。只有当我们new一个对象的时候才会初始化。而第三个参数是用来指明类的加载器的。
    如果查看java.lang.Class类的源代码,上述两种方法最终都会调用Class类中的私有的native方法forName0(),此方法的声明如下:
    private static native Class forName0(String name, boolean init,ClassLoader loader)
    throws ClassNotFoundException;
    所以当我们调用Class.forName(name )时,其实是在方法内部调用了:
    forName0(name, true, ClassLoader.getCallerClassLoader());
    当我们调用Class.forName(name, initialize, loader )的时候,实际上此方法内部调用了:
    forName0(name, initialize, loader);
  


第三十八届CIO班招生
国际CIO认证培训
首席数据官(CDO)认证培训
责编:

免责声明:本网站(http://www.ciotimes.com/)内容主要来自原创、合作媒体供稿和第三方投稿,凡在本网站出现的信息,均仅供参考。本网站将尽力确保所提供信息的准确性及可靠性,但不保证有关资料的准确性及可靠性,读者在使用前请进一步核实,并对任何自主决定的行为负责。本网站对有关资料所引致的错误、不确或遗漏,概不负任何法律责任。
本网站刊载的所有内容(包括但不仅限文字、图片、LOGO、音频、视频、软件、程序等)版权归原作者所有。任何单位或个人认为本网站中的内容可能涉嫌侵犯其知识产权或存在不实内容时,请及时通知本站,予以删除。