那曲檬骨新材料有限公司

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

jvm的類加載器的整體結(jié)構(gòu)及過(guò)程解析

454398 ? 來(lái)源:it610 ? 作者:H.U.C-王子 ? 2020-09-27 15:49 ? 次閱讀

前言

我們很多小伙伴平時(shí)都是做JAVA開(kāi)發(fā)的,那么作為一名合格的工程師,你是否有仔細(xì)的思考過(guò)JVM的運(yùn)行原理呢。

如果懂得了JVM的運(yùn)行原理和內(nèi)存模型,像是一些JVM調(diào)優(yōu)、垃圾回收機(jī)制等等的問(wèn)題我們才能有一個(gè)更清晰的概念。

為了走進(jìn)JVM,深入了解底層,王子打算寫(xiě)一個(gè)JVM的專題,留下自己對(duì)JVM探索的足跡,同時(shí)也希望能幫到小伙伴們更好的理解JVM。

那我們開(kāi)始吧。

JAVA代碼的運(yùn)行流程

首先我們就來(lái)聊一聊JAVA代碼是怎么運(yùn)行起來(lái)的,這部分比較基礎(chǔ)相信大家都知道,就當(dāng)成是個(gè)復(fù)習(xí)吧。

我們編寫(xiě)的代碼都是在java文件中編寫(xiě)的,然后會(huì)編譯成class字節(jié)碼文件。

當(dāng)我們使用到哪個(gè)類的時(shí)候就會(huì)通過(guò)類加載器把class字節(jié)碼文件中的類加載到j(luò)vm內(nèi)存中,然后就是在jvm內(nèi)存中運(yùn)行我們的代碼了。

整體的運(yùn)行流程就是這樣,相信小伙伴們都很清楚這些,但是有關(guān)類加載器是如何把類加載到j(luò)vm內(nèi)存中的,小伙伴們有考慮過(guò)嗎?

今天我們主要就是聊這一部分。

JVM什么時(shí)候加載類

其實(shí)說(shuō)到類加載的底層機(jī)制,這是一個(gè)很復(fù)雜的過(guò)程,但是對(duì)于我們平時(shí)的工作來(lái)講,只要懂得它的核心原理就可以了。

一個(gè)類的加載過(guò)程會(huì)經(jīng)歷如下的幾個(gè)過(guò)程:

加載、驗(yàn)證、準(zhǔn)備、解析、初始化、使用、卸載

首先我們就先弄明白一個(gè)問(wèn)題,jvm是什么時(shí)候去加載類的呢?

其實(shí)答案很簡(jiǎn)單,就是我們什么時(shí)候使用到了這個(gè)類,它就去class字節(jié)碼文件中去加載這個(gè)類。

而作為程序的入口,具有main方法的類,肯定是最開(kāi)始的時(shí)候就加載到j(luò)vm中了。

對(duì)于加載類的時(shí)間點(diǎn)問(wèn)題,其實(shí)就是這么簡(jiǎn)單。

類加載器和雙親委派機(jī)制

既然我們知道了類加載的時(shí)間點(diǎn),那么jvm是通過(guò)什么方式對(duì)類進(jìn)行加載的呢?就是類加載器。

那接下來(lái)我們就來(lái)聊聊jvm的類加載器。

jvm的類加載器總體上可以分成4層,我們一起看一下。

1.啟動(dòng)類加載器

首先就是jvm啟動(dòng)的第一道關(guān)口,啟動(dòng)類加載器Bootstrap ClassLoader,它主要是加載java的核心類。

相信大家都知道,無(wú)論是什么環(huán)節(jié)下運(yùn)行java程序,都是要安裝jvm虛擬機(jī)環(huán)境的,而在這個(gè)環(huán)境的目錄中是有一個(gè)lib文件夾的,這個(gè)文件下就是java最核心的類庫(kù),支撐著java系統(tǒng)的運(yùn)行。

所以一旦jvm啟動(dòng),那么首先就會(huì)通過(guò)啟動(dòng)類加載器去加載lib文件夾下的核心類庫(kù)。

2.擴(kuò)展類加載器

然后我們就到了第二層,擴(kuò)展類加載器Extension ClassLoader,這個(gè)類加載器其實(shí)與啟動(dòng)類加載器是類似的。

在我們的jvm虛擬機(jī)環(huán)境目錄下,是有一個(gè)lib/ext的文件夾的,這里面的類就是java運(yùn)行環(huán)境的一些擴(kuò)展類,這些擴(kuò)展類就是在jvm啟動(dòng)后,通過(guò)擴(kuò)展類加載器進(jìn)行加載的。

3.應(yīng)用程序類加載器

加載完核心類庫(kù)和擴(kuò)展類,這時(shí)候就到了第三層,應(yīng)用程序類加載器Application ClassLoader,這個(gè)類加載器你就可以理解成是加載我們寫(xiě)好的java代碼的就可以了。

4.自定義類加載器

前面的三層就是基本的類加載器了,然后第四層是自定義類加載器,根據(jù)一些特殊的需求來(lái)自己定義類加載器加載我們的類。

整體上類加載器就是這么的4層結(jié)構(gòu)。很多小伙伴可能都聽(tīng)說(shuō)過(guò)雙親委派機(jī)制,那么什么是雙親委派機(jī)制呢,王子就和大家用最接地氣的語(yǔ)言描述一下。

其實(shí)很好理解,就是當(dāng)我們的類加載器要加載一個(gè)類的時(shí)候,它首先會(huì)委派給它的父親去加載,但是如果它的父親沒(méi)找到就會(huì)把這個(gè)事交給他的孩子自己去完成了。

這就是雙親委派機(jī)制。

舉個(gè)例子,假如我們的應(yīng)用程序類加載器要加載一個(gè)類A,那么首先它會(huì)先回家找它老爸?jǐn)U展類加載器,問(wèn)問(wèn)“老爸,你那有這個(gè)類A嗎?”

然后擴(kuò)展類加載器接到這個(gè)請(qǐng)求之后,同樣也懶得處理,再去找它爺爺啟動(dòng)類加載器。

它爺爺找了一圈沒(méi)找到類A,很生氣,就對(duì)擴(kuò)展類加載器說(shuō),“我這沒(méi)有,你自己找去!”

然后擴(kuò)展類加載器就灰溜溜的自己找了一圈,同樣也沒(méi)找到,這時(shí)候就找到應(yīng)用類加載器了,說(shuō):“我這哪有你這個(gè)類A,這明明是你自己應(yīng)該干的活,愛(ài)上哪找上哪找去,我不管了”。

這時(shí)候應(yīng)用類加載器就只能自己去處理了,找了一圈發(fā)現(xiàn)找到了類A,就把它加載到j(luò)vm內(nèi)存中了。

相信大家看了這個(gè)例子應(yīng)該很容易理解了吧。

所以假設(shè)我們自己創(chuàng)建了一個(gè)類java.lang.String,它是不會(huì)被應(yīng)用類加載器加載到內(nèi)存中的,因?yàn)楦割愔锌梢哉业竭@個(gè)類,就直接給加載到內(nèi)存中了。

聊聊驗(yàn)證、準(zhǔn)備、解析、初始化階段

聊完了加載,我們?cè)賮?lái)看看驗(yàn)證、準(zhǔn)備、解析、初始化這幾個(gè)階段jvm都做了什么。

1.驗(yàn)證階段

這一步其實(shí)很容易理解,就是jvm根據(jù)java規(guī)范,來(lái)校驗(yàn)?zāi)慵虞d進(jìn)來(lái)的class文件中的內(nèi)容是否符合規(guī)范,如果不符合規(guī)范jvm是無(wú)法正常運(yùn)行的。

所以在加載后,首先就是驗(yàn)證階段。

2.準(zhǔn)備階段

假設(shè)我們有一個(gè)類A,剛剛加載并通過(guò)了驗(yàn)證,那么就會(huì)進(jìn)行準(zhǔn)備工作。

這個(gè)準(zhǔn)備工作其實(shí)就是給類A分配一定的內(nèi)存空間,然后給里面的靜態(tài)變量(static修飾的變量)也分配內(nèi)存空間,并賦初始值。

3.解析階段

這個(gè)階段干的事實(shí)際上是把符號(hào)引用替換為直接引用,這一過(guò)程網(wǎng)上有很多資料,還是比較復(fù)雜的,如果感興趣小伙伴們可以自己查閱一下資料。

實(shí)際工作中也很少會(huì)接觸這部分的內(nèi)容,所以我們知道有這么個(gè)階段就可以了。

4.初始化階段

在準(zhǔn)備階段,我們把類A的內(nèi)存已經(jīng)分配完了,那么初始化階段要做些什么事呢?我們先看一下類A的代碼

public class A {
    private static String i=System.getProperty("i");
}

準(zhǔn)備階段我們只是給變量i分配了內(nèi)存空間,并賦值了初始值,但是后邊的System.getProperty("i")是不會(huì)執(zhí)行的。

沒(méi)錯(cuò),這部分代碼就是在初始化階段執(zhí)行的,另外靜態(tài)代碼塊也會(huì)在這一階段執(zhí)行。

舉個(gè)例子,比如我們新建一個(gè)對(duì)象new A(),此時(shí)就會(huì)觸發(fā)從加載到初始化的全過(guò)程,把這個(gè)類準(zhǔn)備好并創(chuàng)建一個(gè)實(shí)例對(duì)象。

此外這里有一個(gè)規(guī)則,如果類A繼承了類B,那么在初始化類A的時(shí)候,如果發(fā)現(xiàn)類B還沒(méi)有初始化,會(huì)先初始化類B。

擴(kuò)展

到這里關(guān)于JVM的類加載機(jī)制其實(shí)就已經(jīng)說(shuō)完了,王子再給大家擴(kuò)展一個(gè)小知識(shí)點(diǎn)。

小伙伴們想過(guò)沒(méi)有,Tomcat也是用java開(kāi)發(fā)的,那么它的類加載機(jī)制是什么樣的呢,為什么就能支持jsp呢?

其實(shí)它就是利用了自定義類加載器這一機(jī)制,自己自定義了很多類加載器,整體的結(jié)構(gòu)如下:

Tomcat自定義了這么多的類加載器,用來(lái)加載它自己的核心類庫(kù),并且Tomcat是打破了雙親委派機(jī)制的,感興趣的小伙伴可以自己去查資料了解一下,王子就不在本篇文章長(zhǎng)篇大論來(lái)聊Tomcat了。

總結(jié)

今天我們聊的內(nèi)容還是jvm中比較基礎(chǔ)的部分,以后的文章我們?cè)俾钊耄ヌ剿鱦vm的底層原理,如果對(duì)JVM感興趣的小伙伴可以關(guān)注王子的后續(xù)文章哦,我們可以一步一個(gè)腳印的逐步分解JVM,去了解JVM的垃圾回收機(jī)制、性能調(diào)優(yōu)等等實(shí)用性問(wèn)題,讓你面對(duì)JVM的面試或者生產(chǎn)實(shí)踐也可以游刃有余。

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • JAVA
    +關(guān)注

    關(guān)注

    19

    文章

    2975

    瀏覽量

    105161
  • JVM
    JVM
    +關(guān)注

    關(guān)注

    0

    文章

    158

    瀏覽量

    12262
  • 類加載器
    +關(guān)注

    關(guān)注

    0

    文章

    6

    瀏覽量

    941
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    EtherCAT數(shù)據(jù)幀結(jié)構(gòu)解析

    物理層和常規(guī)的以太網(wǎng)卡,通過(guò)獨(dú)特的數(shù)據(jù)幀結(jié)構(gòu)和處理機(jī)制,實(shí)現(xiàn)了基于EtherNet的實(shí)時(shí)控制。本文將深入探討EtherCAT的數(shù)據(jù)幀結(jié)構(gòu),從幀的組成、子報(bào)文的結(jié)構(gòu)、工作計(jì)數(shù)的功能到數(shù)
    的頭像 發(fā)表于 02-02 17:42 ?186次閱讀

    EE-240: ADSP-BF533 Blackfin加載過(guò)程

    電子發(fā)燒友網(wǎng)站提供《EE-240: ADSP-BF533 Blackfin加載過(guò)程.pdf》資料免費(fèi)下載
    發(fā)表于 01-05 10:00 ?0次下載
    EE-240: ADSP-BF533 Blackfin<b class='flag-5'>加載</b><b class='flag-5'>過(guò)程</b>

    PyTorch 數(shù)據(jù)加載與處理方法

    ,數(shù)據(jù)加載主要依賴于 torch.utils.data 模塊,該模塊提供了 Dataset 和 DataLoader 兩個(gè)核心。 1.1 Dataset Dataset 是 P
    的頭像 發(fā)表于 11-05 17:37 ?497次閱讀

    6網(wǎng)線結(jié)構(gòu)特點(diǎn)有哪些

    網(wǎng)線(CAT6)是網(wǎng)絡(luò)線纜的一種,其結(jié)構(gòu)相對(duì)于五線和超五線來(lái)說(shuō)更為復(fù)雜,以提供更好的傳輸性能和抗干擾能力。以下是六網(wǎng)線
    的頭像 發(fā)表于 09-06 09:58 ?1151次閱讀

    只讀存儲(chǔ)的基本結(jié)構(gòu)和工作過(guò)程

    只讀存儲(chǔ)(Read-Only Memory,ROM)是一種重要的計(jì)算機(jī)存儲(chǔ)設(shè)備,它以非破壞性讀出方式工作,即只能讀出存儲(chǔ)的信息而無(wú)法直接寫(xiě)入新的信息。這種特性使得ROM在存儲(chǔ)固定程序和數(shù)據(jù)方面具有獨(dú)特的優(yōu)勢(shì)。下面將詳細(xì)闡述只讀存儲(chǔ)的基本
    的頭像 發(fā)表于 09-05 10:43 ?1374次閱讀

    從原理聊JVM(一):染色標(biāo)記和垃圾回收算法

    更好地優(yōu)化自己的代碼,并解決一些潛在的性能問(wèn)題。 本文及后續(xù)文章將從原理聊起,對(duì)JVM的內(nèi)存分配、GC、編譯等知識(shí)進(jìn)行分析和總結(jié)。 1 JVM運(yùn)行時(shí)內(nèi)存劃分 1.1 運(yùn)行時(shí)數(shù)據(jù)區(qū)域 ?? ? ? 方法區(qū) 屬于共享內(nèi)存區(qū)域,存儲(chǔ)已被虛擬機(jī)
    的頭像 發(fā)表于 08-20 15:25 ?290次閱讀
    從原理聊<b class='flag-5'>JVM</b>(一):染色標(biāo)記和垃圾回收算法

    labview CAN DBC加載解析程序

    labview CAN DBC加載解析程序
    發(fā)表于 08-18 11:42

    聊聊JVM如何優(yōu)化

    進(jìn)行優(yōu)化。 1.JVM內(nèi)存模型 針對(duì)JAVA8的模型進(jìn)行討論,JVM的內(nèi)存模型主要分為幾個(gè)關(guān)鍵區(qū)域:堆、方法區(qū)、程序計(jì)數(shù)、虛擬機(jī)棧和本地方法棧。堆內(nèi)存進(jìn)一步細(xì)分為年輕代、老年代,年輕代按其特性又分為E區(qū),S1和S2區(qū)。關(guān)于內(nèi)存
    的頭像 發(fā)表于 08-05 17:49 ?528次閱讀
    聊聊<b class='flag-5'>JVM</b>如何優(yōu)化

    卷積神經(jīng)網(wǎng)絡(luò)的基本結(jié)構(gòu)和訓(xùn)練過(guò)程

    處理具有空間層次結(jié)構(gòu)的數(shù)據(jù)時(shí)表現(xiàn)出色。本文將從卷積神經(jīng)網(wǎng)絡(luò)的歷史背景、基本原理、網(wǎng)絡(luò)結(jié)構(gòu)、訓(xùn)練過(guò)程以及應(yīng)用領(lǐng)域等方面進(jìn)行詳細(xì)闡述,以期全面解析這一重要算法。
    的頭像 發(fā)表于 07-02 18:27 ?1058次閱讀

    PLC基本結(jié)構(gòu)解析

    方式和便捷的編程方式,被廣泛應(yīng)用于各種工業(yè)控制系統(tǒng)中。本文將詳細(xì)解析PLC的基本結(jié)構(gòu),包括其主要組成部分的功能和特點(diǎn),以便讀者對(duì)PLC有更深入的了解。
    的頭像 發(fā)表于 06-25 14:30 ?1239次閱讀

    解析經(jīng)典藍(lán)牙設(shè)備連接過(guò)程

    應(yīng)用中,藍(lán)牙設(shè)備發(fā)現(xiàn)、連接、斷開(kāi)等使用場(chǎng)景較為常見(jiàn),其中設(shè)備連接是至關(guān)重要的一環(huán),它涵蓋了設(shè)備之間建立連接的整個(gè)過(guò)程。本文將對(duì)經(jīng)典藍(lán)牙設(shè)備連接過(guò)程進(jìn)行解析,帶大家一起了解連接的整個(gè)過(guò)程
    的頭像 發(fā)表于 06-05 09:11 ?2569次閱讀
    <b class='flag-5'>解析</b>經(jīng)典藍(lán)牙設(shè)備連接<b class='flag-5'>過(guò)程</b>

    并行加載8位寄存數(shù)據(jù)表

    電子發(fā)燒友網(wǎng)站提供《并行加載8位寄存數(shù)據(jù)表.pdf》資料免費(fèi)下載
    發(fā)表于 05-22 09:41 ?0次下載
    并行<b class='flag-5'>加載</b>8位寄存<b class='flag-5'>器</b>數(shù)據(jù)表

    【AWTK使用經(jīng)驗(yàn)】加載和釋放外部圖片

    AWTK是基于C語(yǔ)言開(kāi)發(fā)的跨平臺(tái)GUI框架。《AWTK使用經(jīng)驗(yàn)》系列文章將介紹開(kāi)發(fā)AWTK過(guò)程中一些常見(jiàn)問(wèn)題與解決方案,例如:如何加載外部資源?如何設(shè)計(jì)自定義進(jìn)度條?這些都會(huì)在系列文章進(jìn)行解答。加載
    的頭像 發(fā)表于 04-26 08:25 ?518次閱讀
    【AWTK使用經(jīng)驗(yàn)】<b class='flag-5'>加載</b>和釋放外部圖片

    鴻蒙原生應(yīng)用開(kāi)發(fā)-ArkTS語(yǔ)言基礎(chǔ)庫(kù)概述

    線程之間進(jìn)行通信,開(kāi)發(fā)者需要主動(dòng)創(chuàng)建和關(guān)閉Worker線程。 2.提供常見(jiàn)的容器庫(kù)增、刪、改、查的能力。 3.提供XML、URL、URI構(gòu)造和解析的能力。 XML被設(shè)計(jì)用來(lái)傳輸和存儲(chǔ)數(shù)據(jù),是一種可
    發(fā)表于 03-05 15:42

    ArkTS語(yǔ)言基礎(chǔ)庫(kù)-解析

    多線程并發(fā),支持Worker線程和宿主線程之間進(jìn)行通信,開(kāi)發(fā)者需要主動(dòng)創(chuàng)建和關(guān)閉Worker線程。 提供常見(jiàn)的[容器庫(kù)增、刪、改、查]的能力。 提供XML、URL、URI構(gòu)造和解析的能力。 XML
    發(fā)表于 02-20 16:44
    百家乐高手和勒威| 闲和庄百家乐官网的玩法技巧和规则 | 乐业县| 碧桂园太阳城怎么样| 百家乐真人投注网站| 优博百家乐官网yobo88| 宝马会娱乐城网址| 老虎百家乐的玩法技巧和规则 | 定安县| 开百家乐骗人吗| 赌百家乐官网2号破解| 诚信百家乐官网在线平台| 网络娱乐| 大发888游戏平台hg| 百家乐扎金花斗地主| 租房做生意如何注意风水问题| 百家乐官网游戏下载| 金钱豹娱乐| 宁波水果机遥控器| 粤港澳百家乐娱乐场| 真人百家乐口诀| 百家乐官网公式书| 百家乐官网什么叫缆| 太阳城娱乐城申博| 百家乐棋牌游戏币| 保单百家乐技巧| 百家乐官网tt娱乐场开户注册| 沙龙百家乐官网怎申请| 皇家赌场下载| 德州扑克桌| 战神百家乐的玩法技巧和规则| 澳门百家乐规例| 24山认龙立向| 单机百家乐官网小游戏| 百家乐官网最好打法与投注 | 大发888娱乐游戏--| 百家乐电子路单谁| 百家乐视频游365| 香港百家乐官网玩| 百家乐官网之三姐妹赌博机| 百家乐官网连赢的策略|