JVM 心得分享(加載 鏈接 初始化)
基本概念:類加載的過(guò)程大致分為三個(gè)階段
1、加載階段:本階段主要把class的二進(jìn)制代碼加載進(jìn)入JVM,并且進(jìn)行常量池(類名,方法名,字段名),方法區(qū)(二進(jìn)制字節(jié)碼),棧(本地方法棧結(jié)構(gòu)),堆(java.lang.class對(duì)象)的設(shè)置。
有三個(gè)加載類:Bootstrap ClassLoader,加載jre/lib/下的類;
Extension ClassLoader:加載jre/lib/ext下的類;
ApplicationClassLoader:加載classpath下的類(應(yīng)用程序自己開(kāi)發(fā)的類,如 工程目錄/bin/下的.class文件)
還有一個(gè)擴(kuò)展的加載類,滿足應(yīng)用程序的特殊需求。類的加載時(shí),父親loader優(yōu)先執(zhí)行l(wèi)oad動(dòng)作,父親load不了時(shí),子類運(yùn)作。
2、鏈接階段:又分為三個(gè)小階段 校驗(yàn),準(zhǔn)備,解析。
校驗(yàn):實(shí)施字節(jié)碼文件的格式,語(yǔ)法等的校驗(yàn)。
準(zhǔn)備:對(duì)靜態(tài)變量申請(qǐng)存儲(chǔ)空間,并設(shè)置默認(rèn)的初始值。如:private static int a =2;那么在準(zhǔn)備階段a被設(shè)置為0;
解析:把方法區(qū)中的符號(hào)指針替換為直接引用。
3、初始化階段:對(duì)靜態(tài)變量進(jìn)行初始化,執(zhí)行靜態(tài)塊,創(chuàng)建類的實(shí)例。上述的a變量在初始化階段會(huì)被設(shè)置為2。
第一步:驗(yàn)證靜態(tài)變量和靜態(tài)塊的加載+鏈接(校驗(yàn),準(zhǔn)備,解析)+初始化過(guò)程中值的變化。
package com.chong.studyparalell.clazz.loader;
public class ClassLoaderDemo {
public static void main(String []args){
Test test2 = new Test();
System.out.println("Test2實(shí)例化結(jié)束"+test2.toString());
}
}
package com.chong.studyparalell.clazz.loader;
public class Test{
private static Test test1 = new Test();
private static int a = 2;
private static int b = 2;
static {
System.out.println("【Test類靜態(tài)塊】a=" + a);
}
public Test(){
System.out.println("【Test類構(gòu)造方法】a=" + a);
System.out.println("【Test類構(gòu)造方法】b=" + b);
System.out.println("【Test類實(shí)例】" + this.toString());
}
public static Test newInstance(){
return test1;
}
}
log輸出如下:
1 【Test類構(gòu)造方法】a=0
2 【Test類構(gòu)造方法】b=0
3 【Test類實(shí)例】com.chong.studyparalell.clazz.loader.Test@16c1857
4 【Test類靜態(tài)塊】a=2
5 【Test類構(gòu)造方法】a=2
6 【Test類構(gòu)造方法】b=2
7 【Test類實(shí)例】com.chong.studyparalell.clazz.loader.Test@1b1fd9c
8 Test2實(shí)例化結(jié)束com.chong.studyparalell.clazz.loader.Test@1b1fd9c
首先Test類在鏈接階段(準(zhǔn)備階段),a,b分別被設(shè)置默認(rèn)值0。
當(dāng)new Test()執(zhí)行后,
1)首先初始化Test類的三個(gè)靜態(tài)變量 test1,a,b。
初始化test1時(shí),第一次調(diào)用構(gòu)造方法,此時(shí)a,b為0。對(duì)應(yīng)日志1,2行。
實(shí)例化test1,日志第3行。
test1初始化完成后,繼續(xù)初始化a,b,設(shè)為2。
接著初始化靜態(tài)塊 ,對(duì)應(yīng)日志第4行。
2)執(zhí)行Test類的構(gòu)造方法
因?yàn)閍,b已經(jīng)被初始化為2,所以執(zhí)行類的構(gòu)造方法時(shí),會(huì)輸出a,b 為2。日志第5,6行。
實(shí)例化后輸出地址信息,日志第7行。
3)最終main方法里打出實(shí)例工作完成,日志第8行。
第二步,加入父類后,進(jìn)行確認(rèn)。
package com.chong.studyparalell.clazz.loader;
public class TestBase {
private static int base_a = 2;
private static int base_b = 2;
static {
System.out.println("【父類靜態(tài)塊】 base_a="+base_a);
}
public TestBase(){
System.out.println("【父類 構(gòu)造方法】base_a=" + base_a);
System.out.println("【父類 構(gòu)造方法】base_b=" + base_b);
System.out.println("【父類 實(shí)例】"+ this.toString());
}
}
package com.chong.studyparalell.clazz.loader;
public class Test extends TestBase{
內(nèi)容同第一步;
}
log輸出如下:
【父類靜態(tài)塊】 base_a=2
【父類 構(gòu)造方法】base_a=2
【父類 構(gòu)造方法】base_b=2
【父類 實(shí)例】com.chong.studyparalell.clazz.loader.Test@19ab8d
【Test類構(gòu)造方法】a=0
【Test類構(gòu)造方法】b=0
【Test類實(shí)例】com.chong.studyparalell.clazz.loader.Test@19ab8d
【Test類靜態(tài)塊】a=2
【父類 構(gòu)造方法】base_a=2
【父類 構(gòu)造方法】base_b=2
【父類 實(shí)例】com.chong.studyparalell.clazz.loader.Test@14dcfad
【Test類構(gòu)造方法】a=2
【Test類構(gòu)造方法】b=2
【Test類實(shí)例】com.chong.studyparalell.clazz.loader.Test@14dcfad
Test2實(shí)例化結(jié)束com.chong.studyparalell.clazz.loader.Test@14dcfad
可以發(fā)現(xiàn)父類的靜態(tài)變量,靜態(tài)塊,構(gòu)造方法首先被初始化。然后子類的靜態(tài)變量,靜態(tài)塊和構(gòu)造方法被初始化。
第三步:寫一個(gè)自定義的類加載器
package com.chong.studyparalell.clazz.loader;
public class MyClassLoaderPilot {
public static void main(String[] args) {
try {
MyClassLoader classLoader = new MyClassLoader();
String filename = "com.chong.studyparalell.demon.DemonThreadDemo";
Object clazz = classLoader.loadCustomizeClass(filename);
System.out.println(clazz);
} catch (Exception e) {
e.printStackTrace();
}
}
}
package com.chong.studyparalell.clazz.loader;
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.IOException;
public class MyClassLoader extends ClassLoader {
private String demoPath = "D:\\work\\temp\\";
public Class<?> loadCustomizeClass(String filename) throws ClassNotFoundException {
FileInputStream fis = null;
ByteArrayOutputStream baos = new ByteArrayOutputStream();
try {
// 1.獲取class文件的字節(jié)碼(二進(jìn)制數(shù)據(jù))
String[] fileNames = filename.split("\\.");
fis = new FileInputStream(demoPath + fileNames[fileNames.length-1] +".class");
byte[] bytes = new byte[1024];
int len = 0;
while ((len = fis.read(bytes)) != -1) {
baos.write(bytes, 0, len);
}
} catch (Exception e) {
throw new ClassNotFoundException();
} finally {
try {
fis.close();
} catch (IOException e) {
throw new ClassNotFoundException();
}
}
byte[] paramArrayOfByte = baos.toByteArray();
// 2。把二進(jìn)制文件定義為class對(duì)象返回
return defineClass(filename, paramArrayOfByte, 0, paramArrayOfByte.length);
}
}
日志輸出如下:
class com.chong.studyparalell.demon.DemonThreadDemo
實(shí)際的跟著代碼走一遍,看看控制臺(tái)的輸出,用自己的思路虛擬著跟一跟,對(duì)于類的加載過(guò)程能夠認(rèn)識(shí)的更加清晰一些。
以上這篇JVM 心得分享(加載 鏈接 初始化)就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java中l(wèi)ist.contains()的用法及拓展
List集合相信大家在開(kāi)發(fā)過(guò)程中幾乎都會(huì)用到,有時(shí)候難免會(huì)遇到集合里的數(shù)據(jù)是重復(fù)的,需要進(jìn)行去除,下面這篇文章主要給大家介紹了關(guān)于Java中l(wèi)ist.contains()的用法及拓展的相關(guān)資料,需要的朋友可以參考下2023-03-03
詳解Spring中InitializingBean接口的功能
這篇文章主要介紹了Spring中InitializingBean接口的功能,講述了spring中InitializingBean接口的功能簡(jiǎn)介說(shuō)明,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05
SpringBoot實(shí)戰(zhàn)之實(shí)現(xiàn)結(jié)果的優(yōu)雅響應(yīng)案例詳解
這篇文章主要介紹了SpringBoot實(shí)戰(zhàn)之實(shí)現(xiàn)結(jié)果的優(yōu)雅響應(yīng)案例詳解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09
spring使用ehcache實(shí)現(xiàn)頁(yè)面緩存示例
這篇文章主要介紹了spring使用ehcache實(shí)現(xiàn)頁(yè)面緩存示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-02-02
Java從JDK源碼角度對(duì)Object進(jìn)行實(shí)例分析
這篇文章主要介紹了Java從JDK源碼角度對(duì)Object進(jìn)行實(shí)例分析,具有一定借鑒價(jià)值,需要的朋友可以參考下。2017-12-12
IDEA自定義Maven倉(cāng)庫(kù)的實(shí)現(xiàn)
使用Maven進(jìn)行Java程序開(kāi)發(fā)時(shí),開(kāi)發(fā)者能夠極大地提高開(kāi)發(fā)效率,本文主要介紹了IDEA自定義Maven倉(cāng)庫(kù)的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2024-03-03
深入理解Java中的final關(guān)鍵字_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
Java中的final關(guān)鍵字非常重要,它可以應(yīng)用于類、方法以及變量。這篇文章中我將帶你看看什么是final關(guān)鍵字以及使用final的好處,具體內(nèi)容詳情通過(guò)本文學(xué)習(xí)吧2017-04-04

