suchen

Heim  >  Fragen und Antworten  >  Hauptteil

java程序运行的时候,是把所有的class文件都加载到内存吗?还是用的什么加载什么?

一运行就全部加载?

高洛峰高洛峰2806 Tage vor599

Antworte allen(4)Ich werde antworten

  • PHP中文网

    PHP中文网2017-04-18 10:51:58

    并不是全部加载,可以理解为按需加载。
    比如继承的父类,实现的接口,声明的变量,方法的返回类型等。

    就是在使用当前的类时,缺少哪些必要的类都会立即加载。

    Antwort
    0
  • 天蓬老师

    天蓬老师2017-04-18 10:51:58

    按需加载。
    类加载的时机:

    1)遇到new、getstatic、putstatic或invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。生成这4条指令的最常见的Java代码场景是:使用new关键字实例化对象的时候、读取或设置一个类的静态字段(被final修饰、已在编译期把结果放入常量池的静态字段除外)的时候,以及调用一个类的静态方法的时候。
    2)使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。
    3)当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。
    4)当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。
    5)当使用JDK 1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。
    对于这5种会触发类进行初始化的场景,虚拟机规范中使用了一个很强烈的限定语:“有且只有”,这5种场景中的行为称为对一个类进行主动引用。除此之外,所有引用类的方式都不会触发初始化,称为被动引用。

    来源:深入理解JVM 7.2 类加载的时机

    Antwort
    0
  • 巴扎黑

    巴扎黑2017-04-18 10:51:58

    这一块还没有深入了解,不敢误人子弟。
    我知道的,虚拟机在运行的时候,会预先加载一个常用的class,比如java.lang包下面的。
    至于你在程序中自己引用的class文件/jar包之类的,是有一个加载过程的。
    当虚拟机需要一个class文件,他会调用默认的classLoader.loadClass()去加载,这个方法首先在已经加载的class中查找,找得到当然返回,找不到则交给父加载器。层层查找没有之后,classLoader会调用findClass方法去文件中加载。
    protected Class<?> loadClass(String name, boolean resolve)

        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }
    
                if (c == null) {
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    long t1 = System.nanoTime();
                    c = findClass(name);
    
                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }

    Antwort
    0
  • 伊谢尔伦

    伊谢尔伦2017-04-18 10:51:58

    http://blog.csdn.net/briblue/...
    可以参考这篇博客 讲的不错

    Antwort
    0
  • StornierenAntwort