JVM:JVMクラスローダClassLoaderの実装-コード角度
2691 ワード
次にClassLoaderでclassをロードする方法を示します.
1、同期ロックとロックにより、同じclassに一度だけロードすることを保証する2、親ローダを先に検索し、1階1階上を探し、nullであれば、最上位を見つけたことを説明し、最終的な親ローダにロードを依頼する.3.親がロードできない場合は、子がローダにロードされていることをさらに伝えます.4、クラスのロードに成功した後、resolveがtrueに転送されたかどうかを判断する必要があり、転送trueはクラス参照をロードする必要がある他のクラスを表す.クラスローダの戻り間にすべてロードする必要があります.
次に、コード・ケースを使用して、自分で書いたクラスの下でどのクラス・ローダを使用しているかを見てみましょう.
どちらの方法でも、アプリケーション・クラス・ローダを取得できます.結果:
注意AppClassLoaderはclasspathの下のクラスをロードします
ちなみにAppClassLoaderの親ローダーを見てみましょう.
結果:
AppClassLoaderの親ローダーは、ExtClassLoader、拡張クラスローダーであることがわかります.注意:ExtClassLoaderロード/lib/extの下のクラス
ExtClassLoaderの親ローダを見てみましょう:結果はnullです.ExtClassLoaderの親ローダはBootstrapClassLoaderですが、この起動クラスローダはC++で実装されており、JVMの一部であるため、JVMのメモリにはこのオブジェクトが見つかりません.
protected Class> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) { ---1
// First, check if the class has already been loaded
Class c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
--2
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
}
--3
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();
}
}
--4
if (resolve) {
resolveClass(c);
}
return c;
}
}
1、同期ロックとロックにより、同じclassに一度だけロードすることを保証する2、親ローダを先に検索し、1階1階上を探し、nullであれば、最上位を見つけたことを説明し、最終的な親ローダにロードを依頼する.3.親がロードできない場合は、子がローダにロードされていることをさらに伝えます.4、クラスのロードに成功した後、resolveがtrueに転送されたかどうかを判断する必要があり、転送trueはクラス参照をロードする必要がある他のクラスを表す.クラスローダの戻り間にすべてロードする必要があります.
次に、コード・ケースを使用して、自分で書いたクラスの下でどのクラス・ローダを使用しているかを見てみましょう.
System.out.println(ClassLoader.getSystemClassLoader());
System.out.println(BackendController.class.getClassLoader());
どちらの方法でも、アプリケーション・クラス・ローダを取得できます.結果:
sun.misc.Launcher$AppClassLoader@7f423820
sun.misc.Launcher$AppClassLoader@7f423820
注意AppClassLoaderはclasspathの下のクラスをロードします
ちなみにAppClassLoaderの親ローダーを見てみましょう.
System.out.println(ClassLoader.getSystemClassLoader().getParent());
結果:
sun.misc.Launcher$ExtClassLoader@5472fe25
AppClassLoaderの親ローダーは、ExtClassLoader、拡張クラスローダーであることがわかります.注意:ExtClassLoaderロード/lib/extの下のクラス
ExtClassLoaderの親ローダを見てみましょう:結果はnullです.ExtClassLoaderの親ローダはBootstrapClassLoaderですが、この起動クラスローダはC++で実装されており、JVMの一部であるため、JVMのメモリにはこのオブジェクトが見つかりません.