国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語(yǔ)言|JavaScript|易語(yǔ)言|vb.net|

香港云服务器
服務(wù)器之家 - 編程語(yǔ)言 - Java教程 - 關(guān)于Java虛擬機(jī)HotSpot

關(guān)于Java虛擬機(jī)HotSpot

2022-03-10 00:52鳩摩 Java教程

這篇文章主要介紹了關(guān)于Java虛擬機(jī)HotSpot,在Java類中的一些方法會(huì)被由C/C++編寫的HotSpot虛擬機(jī)的C/C++函數(shù)調(diào)用,不過(guò)由于Java方法與C/C++函數(shù)的調(diào)用約定不同,所以并不能直接調(diào)用,需要JavaCalls::call()這個(gè)函數(shù)輔助調(diào)用,下面我們來(lái)看

我們寫的主類中的main()方法是如何被Java虛擬機(jī)調(diào)用到的?在Java類中的一些方法會(huì)被由C/C++編寫的HotSpot虛擬機(jī)的C/C++函數(shù)調(diào)用,不過(guò)由于Java方法與C/C++函數(shù)的調(diào)用約定不同,所以并不能直接調(diào)用,需要JavaCalls::call()這個(gè)函數(shù)輔助調(diào)用。(我把由C/C++編寫的叫函數(shù),把Java編寫的叫方法,后續(xù)也會(huì)延用這樣的叫法)如下圖所示。

關(guān)于Java虛擬機(jī)HotSpot

從C/C++函數(shù)中調(diào)用的一些Java方法主要有:

  • (1)Java主類中的main()方法;
  • (2)Java主類裝載時(shí),調(diào)用JavaCalls::call()函數(shù)執(zhí)行checkAndLoadMain()方法;
  • (3)類的初始化過(guò)程中,調(diào)用JavaCalls::call()函數(shù)執(zhí)行的Java類初始化方法<clinit>,可以查看JavaCalls::call_default_constructor()函數(shù),有對(duì)<clinit>方法的調(diào)用邏輯;
  • (4)我們先省略main方法的執(zhí)行流程(其實(shí)main方法的執(zhí)行也是先啟動(dòng)一個(gè)JavaMain線程,套路都是一樣的),單看某個(gè)JavaThread的啟動(dòng)過(guò)程。JavaThread的啟動(dòng)最終都要通過(guò)一個(gè)native方法java.lang.Thread#start0()方法完成的,這個(gè)方法經(jīng)過(guò)解釋器的native_entry入口,調(diào)用到了JVM_StartThread()函數(shù)。其中的static void thread_entry(JavaThread* thread, TRAPS)函數(shù)中會(huì)調(diào)用JavaCalls::call_virtual()函數(shù)。JavaThread最終會(huì)通過(guò)JavaCalls::call_virtual()函數(shù)來(lái)調(diào)用字節(jié)碼中的run()方法;
  • (5)在SystemDictionary::load_instance_class()這個(gè)能體現(xiàn)雙親委派的函數(shù)中,如果類加載器對(duì)象不為空,則會(huì)調(diào)用這個(gè)類加載器的loadClass()函數(shù)(通過(guò)call_virtual()函數(shù)來(lái)調(diào)用)來(lái)加載類。

當(dāng)然還會(huì)有其它方法,這里就不一一列舉了。通過(guò)JavaCalls::call() 、JavaCalls::call_helper()等函數(shù)調(diào)用Java方法,這些函數(shù)定義在JavaCalls類中,

這個(gè)類的定義如下:

從C/C++函數(shù)中調(diào)用的一些Java方法主要有:

  • (1)Java主類中的main()方法;
  • (2)Java主類裝載時(shí),調(diào)用JavaCalls::call()函數(shù)執(zhí)行checkAndLoadMain()方法;
  • (3)類的初始化過(guò)程中,調(diào)用JavaCalls::call()函數(shù)執(zhí)行的Java類初始化方法<clinit>,可以查看JavaCalls::call_default_constructor()函數(shù),有對(duì)<clinit>方法的調(diào)用邏輯;
  • (4)我們先省略main方法的執(zhí)行流程(其實(shí)main方法的執(zhí)行也是先啟動(dòng)一個(gè)JavaMain線程,套路都是一樣的),單看某個(gè)JavaThread的啟動(dòng)過(guò)程。JavaThread的啟動(dòng)最終都要通過(guò)一個(gè)native方法java.lang.Thread#start0()方法完成的,這個(gè)方法經(jīng)過(guò)解釋器的native_entry入口,調(diào)用到了JVM_StartThread()函數(shù)。其中的static void thread_entry(JavaThread* thread, TRAPS)函數(shù)中會(huì)調(diào)用JavaCalls::call_virtual()函數(shù)。JavaThread最終會(huì)通過(guò)JavaCalls::call_virtual()函數(shù)來(lái)調(diào)用字節(jié)碼中的run()方法;
  • (5)在SystemDictionary::load_instance_class()這個(gè)能體現(xiàn)雙親委派的函數(shù)中,如果類加載器對(duì)象不為空,則會(huì)調(diào)用這個(gè)類加載器的loadClass()函數(shù)(通過(guò)call_virtual()函數(shù)來(lái)調(diào)用)來(lái)加載類。

當(dāng)然還會(huì)有其它方法,這里就不一一列舉了。通過(guò)JavaCalls::call()JavaCalls::call_helper()等函數(shù)調(diào)用Java方法,這些函數(shù)定義在JavaCalls類中,

這個(gè)類的定義如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
源代碼位置:openjdk/hotspot/src/share/vm/runtime/javaCalls.hpp
 
class JavaCalls: AllStatic {
  static void call_helper(JavaValue* result, methodHandle* method, JavaCallArguments* args, TRAPS);
 public:
  
  static void call_default_constructor(JavaThread* thread, methodHandle method, Handle receiver, TRAPS);
 
  // 使用如下函數(shù)調(diào)用Java中一些特殊的方法,如類初始化方法<clinit>等
  // receiver表示方法的接收者,如A.main()調(diào)用中,A就是方法的接收者
  static void call_special(JavaValue* result, KlassHandle klass, Symbol* name,Symbol* signature, JavaCallArguments* args, TRAPS);
  static void call_special(JavaValue* result, Handle receiver, KlassHandle klass,Symbol* name, Symbol* signature, TRAPS);
  static void call_special(JavaValue* result, Handle receiver, KlassHandle klass,Symbol* name, Symbol* signature, Handle arg1, TRAPS);
  static void call_special(JavaValue* result, Handle receiver, KlassHandle klass,Symbol* name, Symbol* signature, Handle arg1, Handle arg2, TRAPS);
 
  // 使用如下函數(shù)調(diào)用動(dòng)態(tài)分派的一些方法
  static void call_virtual(JavaValue* result, KlassHandle spec_klass, Symbol* name,Symbol* signature, JavaCallArguments* args, TRAPS);
  static void call_virtual(JavaValue* result, Handle receiver, KlassHandle spec_klass,Symbol* name, Symbol* signature, TRAPS);
  static void call_virtual(JavaValue* result, Handle receiver, KlassHandle spec_klass,Symbol* name, Symbol* signature, Handle arg1, TRAPS);
  static void call_virtual(JavaValue* result, Handle receiver, KlassHandle spec_klass,Symbol* name, Symbol* signature, Handle arg1, Handle arg2, TRAPS);
 
  // 使用如下函數(shù)調(diào)用Java靜態(tài)方法
  static void call_static(JavaValue* result, KlassHandle klass,Symbol* name, Symbol* signature, JavaCallArguments* args, TRAPS);
   static void call_static(JavaValue* result, KlassHandle klass,Symbol* name, Symbol* signature, TRAPS);
  static void call_static(JavaValue* result, KlassHandle klass,Symbol* name, Symbol* signature, Handle arg1, TRAPS);
  static void call_static(JavaValue* result, KlassHandle klass,Symbol* name, Symbol* signature, Handle arg1, Handle arg2, TRAPS);
 
  // 更低一層的接口,如上的一些函數(shù)可能會(huì)最終調(diào)用到如下這個(gè)函數(shù)
  static void call(JavaValue* result, methodHandle method, JavaCallArguments* args, TRAPS);
};

如上的函數(shù)都是自解釋的,通過(guò)名稱我們就能看出這些函數(shù)的作用。其中JavaCalls::call()函數(shù)是更低一層的通用接口。Java虛擬機(jī)規(guī)范定義的字節(jié)碼指令共有5個(gè),分別為invokestatic 、invokedynamic 、invokestatic 、invokespecial、invokevirtual幾種方法調(diào)用指令。這些call_static() 、call_virtual()函數(shù)內(nèi)部調(diào)用了call()函數(shù)。這一節(jié)我們先不介紹各個(gè)方法的具體實(shí)現(xiàn)。下一篇將詳細(xì)介紹。

我們選一個(gè)重要的main()方法來(lái)查看具體的調(diào)用邏輯。如下基本照搬R大的內(nèi)容,不過(guò)我略做了一些修改,如下:

假設(shè)我們的Java主類的類名為JavaMainClass,下面為了區(qū)分java launcher里C/C++的main()Java層程序里的main(),把后者寫作JavaMainClass.main()方法。

從剛進(jìn)入C/C++的main()函數(shù)開始:

啟動(dòng)并調(diào)用HotSpot虛擬機(jī)的main()函數(shù)的線程執(zhí)行的主要邏輯如下:

?
1
2
3
main()
-> //... 做一些參數(shù)檢查
-> //... 開啟新線程作為main線程,讓它從JavaMain()函數(shù)開始執(zhí)行;該線程等待main線程執(zhí)行結(jié)束

在如上線程中會(huì)啟動(dòng)另外一個(gè)線程執(zhí)行JavaMain()函數(shù),如下:

?
1
2
3
4
5
6
JavaMain()
-> //... 找到指定的JVM
-> //... 加載并初始化JVM
-> //... 根據(jù)Main-Class指定的類名加載JavaMainClass
-> //... 在JavaMainClass類里找到名為"main"的方法,簽名為"([Ljava/lang/String;)V",修飾符是public的靜態(tài)方法
-> (*env)->CallStaticVoidMethod(env, mainClass, mainID, mainArgs); // 通過(guò)JNI調(diào)用JavaMainClass.main()方法

以上步驟都還在java launcher的控制下;當(dāng)控制權(quán)轉(zhuǎn)移到JavaMainClass.main()方法之后就沒(méi)java launcher什么事了,等JavaMainClass.main()方法返回之后java launcher才接手過(guò)來(lái)清理和關(guān)閉JVM。

下面看一下調(diào)用Java主類main()方法時(shí)會(huì)經(jīng)過(guò)的主要方法及執(zhí)行的主要邏輯,如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
// HotSpot VM里對(duì)JNI的CallStaticVoidMethod的實(shí)現(xiàn)。留意要傳給Java方法的參數(shù)
// 以C的可變長(zhǎng)度參數(shù)傳入,這個(gè)函數(shù)將其收集打包為JNI_ArgumentPusherVaArg對(duì)象
-> jni_CallStaticVoidMethod() 
 
     // 這里進(jìn)一步將要傳給Java的參數(shù)轉(zhuǎn)換為JavaCallArguments對(duì)象傳下去   
     -> jni_invoke_static() 
       
        // 真正底層實(shí)現(xiàn)的開始。這個(gè)方法只是層皮,把JavaCalls::call_helper()
        // 用os::os_exception_wrapper()包裝起來(lái),目的是設(shè)置HotSpot VM的C++層面的異常處理
        -> JavaCalls::call()  
     
           -> JavaCalls::call_helper()
              -> //... 檢查目標(biāo)方法是否為空方法,是的話直接返回
              -> //... 檢查目標(biāo)方法是否“首次執(zhí)行前就必須被編譯”,是的話調(diào)用JIT編譯器去編譯目標(biāo)方法
              -> //... 獲取目標(biāo)方法的解釋模式入口from_interpreted_entry,下面將其稱為entry_point
              -> //... 確保Java棧溢出檢查機(jī)制正確啟動(dòng)
              -> //... 創(chuàng)建一個(gè)JavaCallWrapper,用于管理JNIHandleBlock的分配與釋放,
                 // 以及在調(diào)用Java方法前后保存和恢復(fù)Java的frame pointer/stack pointer
 
              //... StubRoutines::call_stub()返回一個(gè)指向call stub的函數(shù)指針,
              // 緊接著調(diào)用這個(gè)call stub,傳入前面獲取的entry_point和要傳給Java方法的參數(shù)等信息
              -> StubRoutines::call_stub()(...)
                 // call stub是在VM初始化時(shí)生成的。對(duì)應(yīng)的代碼在
                 // StubGenerator::generate_call_stub()函數(shù)中
                 -> //... 把相關(guān)寄存器的狀態(tài)調(diào)整到解釋器所需的狀態(tài)
                 -> //... 把要傳給Java方法的參數(shù)從JavaCallArguments對(duì)象解包展開到解釋模
                    // 式calling convention所要求的位置
                 -> //... 跳轉(zhuǎn)到前面?zhèn)魅氲膃ntry_point,也就是目標(biāo)方法的from_interpreted_entry
 
                    -> //... 在-Xcomp模式下,實(shí)際跳入的是i2c adapter stub,將解釋模式calling convention
                       // 傳入的參數(shù)挪到編譯模式calling convention所要求的位置
                           -> //... 跳轉(zhuǎn)到目標(biāo)方法被JIT編譯后的代碼里,也就是跳到 nmethod 的 VEP 所指向的位置
                                -> //... 正式開始執(zhí)行目標(biāo)方法被JIT編譯好的代碼 <- 這里就是"main()方法的真正入口"

后面3個(gè)步驟是在編譯執(zhí)行的模式下,不過(guò)后續(xù)我們從解釋執(zhí)行開始研究,所以需要為虛擬機(jī)配置-Xint選項(xiàng),有了這個(gè)選項(xiàng)后,Java主類的main()方法就會(huì)解釋執(zhí)行了。

在調(diào)用Java主類main()方法的過(guò)程中,我們看到了虛擬機(jī)是通過(guò)JavaCalls::call()函數(shù)來(lái)間接調(diào)用main()方法的,下一篇我們研究一下具體的調(diào)用邏輯。

到此這篇關(guān)于關(guān)于Java虛擬機(jī)HotSpot的文章就介紹到這了,更多相關(guān)Java虛擬機(jī)HotSpot內(nèi)容請(qǐng)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!

原文鏈接:https://www.heapdump.cn/article/2860175

延伸 · 閱讀

精彩推薦
  • Java教程java中Path和ClassPath用法比較

    java中Path和ClassPath用法比較

    在本篇文章里小編給大家分享了關(guān)于java中Path和ClassPath用法比較內(nèi)容,有需要的朋友們學(xué)習(xí)下。...

    Java之家12082021-07-12
  • Java教程Spring Boot JPA如何把ORM統(tǒng)一起來(lái)

    Spring Boot JPA如何把ORM統(tǒng)一起來(lái)

    Spring Data JPA 是 Spring 基于 ORM 框架、JPA 規(guī)范的基礎(chǔ)上封裝的一套JPA應(yīng)用框架,可使開發(fā)者用極簡(jiǎn)的代碼即可實(shí)現(xiàn)對(duì)數(shù)據(jù)的訪問(wèn)和操作,本文給大家詳細(xì)介紹了...

    張占嶺8202021-04-22
  • Java教程JAVA利用泛型返回類型不同的對(duì)象方法

    JAVA利用泛型返回類型不同的對(duì)象方法

    下面小編就為大家?guī)?lái)一篇JAVA利用泛型返回類型不同的對(duì)象方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧...

    jingxian5982020-08-15
  • Java教程Java中常見(jiàn)的5種WEB服務(wù)器介紹

    Java中常見(jiàn)的5種WEB服務(wù)器介紹

    這篇文章主要介紹了Java中常見(jiàn)的5種WEB服務(wù)器介紹,它們分別是Tomcat、Resin、JBoss、WebSphere、WebLogic,需要的朋友可以參考下 ...

    junjie3882019-11-24
  • Java教程solr在java中的使用實(shí)例代碼

    solr在java中的使用實(shí)例代碼

    本篇文章主要介紹了solr在java中的使用實(shí)例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧...

    俺就不起網(wǎng)名4452020-11-25
  • Java教程解決java idea新建子目錄時(shí)命名不是樹形結(jié)構(gòu)的問(wèn)題

    解決java idea新建子目錄時(shí)命名不是樹形結(jié)構(gòu)的問(wèn)題

    這篇文章主要介紹了解決java idea新建子目錄時(shí)命名不是樹形結(jié)構(gòu)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧 ...

    光哥_帥3682020-08-17
  • Java教程Java 數(shù)組分析及簡(jiǎn)單實(shí)例

    Java 數(shù)組分析及簡(jiǎn)單實(shí)例

    這篇文章主要介紹了Java 數(shù)組分析及簡(jiǎn)單實(shí)例的相關(guān)資料,在Java中它就是對(duì)象,一個(gè)比較特殊的對(duì)象,需要的朋友可以參考下...

    Java之家2282020-08-23
  • Java教程Java8 中的ParallelStream

    Java8 中的ParallelStream

    這篇文章主要介紹了Java8 中的并行流 ParallelStreams,Java8并行流ParallelStream和Stream的區(qū)別就是支持并行執(zhí)行,提高程序運(yùn)行效率。下面就來(lái)看看文章內(nèi)容具體...

    onlythinking6232022-02-24
1019
主站蜘蛛池模板: 午夜在线观看视频 | 国产 欧美 日韩 一区 | 色视频免费在线观看 | 欧美综合一区 | 国产99久久久精品视频 | 玖玖在线免费视频 | 美日韩一区二区 | 欧美国产日韩视频 | 四虎影| 国产精品高清在线 | 欧美福利视频 | 久久精品一区二区国产 | 久久99这里只有精品 | 亚洲精品永久免费 | 亚洲免费精品 | 欧美日韩国产一区二区三区 | 日韩欧美国产一区二区 | 精品欧美| 情一色一乱一欲一区二区 | av一级久久 | 国产综合网站 | 国产成人在线一区 | 亚洲精品乱码久久久久久久 | 天天操人人干 | 国产色网 | 久久精品久久久久久久久久16 | 高清av在线 | 中文字幕在线观看1 | 亚洲第一福利视频 | 久久精品夜夜夜夜夜久久 | 伊人二区| 五月天导航 | 人和拘一级毛片 | 午夜免费视频网站 | 免费又黄又爽又猛的毛片 | 免费观看视频毛片 | av三级在线观看 | 黄视频免费观看网站 | 欧美精品成人一区二区三区四区 | 懂色av中文字幕一区二区三区 | 久久国内 |