java程序運行的時候,是把所有的class文件都加載到內存嗎?還是用的什么加載什么?
問題描述
一運行就全部加載?
問題解答
回答1:并不是全部加載,可以理解為按需加載。比如繼承的父類,實現的接口,聲明的變量,方法的返回類型等。
就是在使用當前的類時,缺少哪些必要的類都會立即加載。
回答2:按需加載。類加載的時機:
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 類加載的時機
回答3:這一塊還沒有深入了解,不敢誤人子弟。我知道的,虛擬機在運行的時候,會預先加載一個常用的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 loadedClass<?> 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 statssun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);sun.misc.PerfCounter.getFindClasses().increment(); }}if (resolve) { resolveClass(c);}return c; }回答4:
http://blog.csdn.net/briblue/...可以參考這篇博客 講的不錯
