java基礎(chǔ)學(xué)習(xí)筆記之類加載器
類加載器
java類加載器就是在運(yùn)行時(shí)在JVM中動(dòng)態(tài)地加載所需的類,java類加載器基于三個(gè)機(jī)制:委托,可見,單一。
把classpath下的那些.class文件加載進(jìn)內(nèi)存,處理后成為字節(jié)碼,這些工作是類加載器做的。
- 委托機(jī)制指的是將加載類的請(qǐng)求傳遞給父加載器,如果父加載器找不到或者不能加載這個(gè)類,那么再加載他。
- 可見性機(jī)制指的是父加載器加載的類都能被子加載器看見,但是子加載器加載的類父加載器是看不見的。
- 單一性機(jī)制指的是一個(gè)類只能被同一種加載器加載一次。
默認(rèn)類加載器
系統(tǒng)默認(rèn)三個(gè)類加載器:
- BootStrap
- ExtClassLoader
- AppClassLoader
類加載器也是java類,而BootStrap不是。 驗(yàn)證代碼:
public class ClassLoaderTest {
public static void main(String[] args) {
System.out.println(System.class.getClassLoader());
}
}
輸出:null
如果使用System.out.println(System.class.getClassLoader().toString);,則報(bào)空指針異常:
Exception in thread "main" java.lang.NullPointerException at com.iot.classloader.ClassLoaderTest.main(ClassLoaderTest.java:10) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.lang.reflect.Method.invoke(Method.java:483) at com.intellij.rt.execution.application.AppMain.main(AppMain.java:144)
可見,System類是由BootStrap類加載器加載。
類加載器的委托機(jī)制
類加載器的樹狀圖
類加載器
一般加載類的順序:
- 首先當(dāng)前線程的類加載器去加載線程中的第一個(gè)類
- 如果類A應(yīng)用了類B,java虛擬機(jī)將使用加載類A的類加載器來加載類B
- 還可以直接調(diào)用ClassLoader.loadClass()方法來制定某個(gè)類加載器去加載某個(gè)類
自定義類加載器的編寫原理
API:
Class ClassLoader
模板方法設(shè)計(jì)模式
父類:
loadClass(類加載的流程,模板)
findClass供子類覆蓋的、被loadClass方法調(diào)用的類加載邏輯
defineClass得到class文件轉(zhuǎn)換成字節(jié)碼
子類:覆蓋findClass方法
例子:
loadClass方法的源碼
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;
}
}
API文檔中的例子:
class NetworkClassLoader extends ClassLoader {
String host;
int port;
public Class findClass(String name) {
byte[] b = loadClassData(name);
return defineClass(name, b, 0, b.length);
}
private byte[] loadClassData(String name) {
// load the class data from the connection
. . .
}
}
相關(guān)文章
SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息
在一次項(xiàng)目開發(fā)中,使用到了Netty網(wǎng)絡(luò)應(yīng)用框架,以及MQTT進(jìn)行消息數(shù)據(jù)的收發(fā),這其中需要后臺(tái)來將獲取到的消息主動(dòng)推送給前端,所以本文記錄了SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的操作,需要的朋友可以參考下2024-02-02
Java中dubbo+zookeeper微服務(wù)架構(gòu)簡(jiǎn)介
Apache Dubbo是一款高性能的 Java RPC 框架,這篇文章主要介紹了Java中dubbo+zookeeper微服務(wù)架構(gòu),需要的朋友可以參考下2021-09-09
Spring整合Quartz Job以及Spring Task的實(shí)現(xiàn)方法
下面小編就為大家分享一篇Spring整合Quartz Job以及Spring Task的實(shí)現(xiàn)方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2017-12-12

