溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何解析JVM 字節(jié)碼指令

發(fā)布時(shí)間:2021-10-23 16:04:24 來源:億速云 閱讀:133 作者:柒染 欄目:大數(shù)據(jù)

如何解析JVM 字節(jié)碼指令,相信很多沒有經(jīng)驗(yàn)的人對(duì)此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個(gè)問題。

  在平時(shí)的demo中,依靠字節(jié)碼順序,解析程序執(zhí)行流程,真正的執(zhí)行順序是字節(jié)碼的執(zhí)行順序,單線程下字節(jié)碼順序是與程序書寫順序一致的,多線程環(huán)境下,共享變量的賦值讀取順序卻不可掌握時(shí)機(jī)。

   JVM:Java Virtual Machine 

如何解析JVM 字節(jié)碼指令

ps:這是我搜集的最干凈整潔的JVM內(nèi)存圖了

MinorGC的過程(復(fù)制->清空->互換)

1:Eden,SurvivorFrom復(fù)制到SurvivorTo,年齡+1

首先,當(dāng)Eden區(qū)滿的時(shí)候會(huì)觸發(fā)第一次GC,把還活著的對(duì)象拷貝到SurvivorFrom區(qū),當(dāng)Eden區(qū)再次出發(fā)GC的時(shí)候會(huì)掃描Eden區(qū)和form區(qū),對(duì)這個(gè)區(qū)域進(jìn)行垃圾回收,經(jīng)過這次回收還活著的,復(fù)制到To區(qū),對(duì)象年齡+1

2:清空Eden區(qū)、SurvivorFrom

然后清空Eden區(qū)和SurvivorFrom區(qū)的對(duì)象,誰空誰是to。

3:SurvivorTo和SurvivorFrom互換

互換之后SurvivorTo成為下一次GC的From區(qū),當(dāng)對(duì)象年齡達(dá)到15,最終如果存活,存入老年代。

GC算法

標(biāo)記清除算法:先標(biāo)記回收對(duì)象,再統(tǒng)一回收

如何解析JVM 字節(jié)碼指令

標(biāo)記壓縮,標(biāo)記清除后,將非連續(xù)空間進(jìn)行壓縮

如何解析JVM 字節(jié)碼指令

復(fù)制算法

把空間分成兩塊,每次只對(duì)其中一塊進(jìn)行 GC。當(dāng)這塊內(nèi)存使用完時(shí),就將還存活的對(duì)象復(fù)制到另一塊上面。

引用計(jì)數(shù)法:循環(huán)引用不可回收,不推薦

GCRoot:可達(dá)性分析算法

從根集對(duì)象向下搜索,如果一個(gè)對(duì)象沒有任何鏈相連時(shí),則說明對(duì)象不可用。

哪些可以作為GC root的對(duì)象

  1. 虛擬機(jī)棧中的引用對(duì)象

  2. 方法區(qū)中的類靜態(tài)屬性引用的對(duì)象

  3. 方法去中常量引用的對(duì)象

  4. 本地方法棧中引用的對(duì)象

如何確定垃圾?

已經(jīng)不再被內(nèi)存使用到的空間

JVM 參數(shù)

JVM 系統(tǒng)默認(rèn)值Xms Xmx  做好調(diào)成一致  避免GC頻繁收集  忽高忽低

XX類型:boolean類型,KV設(shè)值類型,jinfo類型

+-表示是否開啟

-XX:+PrintGCDetails

-XX:+UseSerialGC

如何解析JVM 字節(jié)碼指令

Heap PSYoungGen      total 38400K, used 4366K [0x00000000d5a00000, 0x00000000d8480000, 0x0000000100000000)  eden space 33280K, 10% used [0x00000000d5a00000,0x00000000d5d89bd0,0x00000000d7a80000)  from space 5120K, 14% used [0x00000000d7f80000,0x00000000d803a020,0x00000000d8480000)  to   space 5120K, 0% used [0x00000000d7a80000,0x00000000d7a80000,0x00000000d7f80000) ParOldGen       total 87552K, used 16K [0x0000000080e00000, 0x0000000086380000, 0x00000000d5a00000)  object space 87552K, 0% used [0x0000000080e00000,0x0000000080e04000,0x0000000086380000)str=  kkget Metaspace       used 3352K, capacity 4556K, committed 4864K, reserved 1056768K
 class space    used 355K, capacity 392K, committed 512K, reserved 1048576

-Xms:初始內(nèi)存大小默認(rèn)為物理內(nèi)存的1/64

-Xmx:最大內(nèi)存大小,默認(rèn)為物理內(nèi)存的1/4

-Xss:?jiǎn)蝹€(gè)線程的大小,一般為512k-1024k

如何解析JVM 字節(jié)碼指令

-Xmn:設(shè)置年輕代大小

-XX:MetespaceSize:設(shè)置元空間大小,元空間使用本地內(nèi)存

垃圾收集器: 并行 串行 并發(fā)標(biāo)記 G1 ZGC

1.串行垃圾回收器(Serial)單線程環(huán)境設(shè)計(jì)只用一個(gè)線程回收

2.并行垃圾回收器(Parellel)多個(gè)收集線程并行工作

3.并發(fā)垃圾回收器(CMS)用戶線程和垃圾回收線程同事執(zhí)行

4.G1垃圾回收器 使用堆內(nèi)存很大的情況,分割區(qū)域回收,java8

如何查看服務(wù)器默認(rèn)垃圾回收器?

-XX:+PrintFlagsFinal || -XX:+PrintCommandLineFlags

bool UseSerialGC   := true  {product}-XX:InitialHeapSize=133236224 -XX:MaxHeapSize=2131779584 -XX:+PrintCommandLineFlags -XX:+PrintFlagsFinal -XX:+PrintGCDetails -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:-UseLargePagesIndividualAllocation -XX:+UseSerialGC

如何解析JVM 字節(jié)碼指令

G1不產(chǎn)生內(nèi)存碎片 可精準(zhǔn)控制停頓

字節(jié)碼指令解析

以Price問題為例

package com.kk;
import org.junit.Assert;import org.junit.Before;import org.junit.Test;import org.junit.jupiter.api.AfterEach;import org.junit.jupiter.api.BeforeEach;
import java.util.ArrayList;
public class Price {
   public static final Price INSTANCE = new Price(12);
   private static volatile int staticPrice = 5;
   public int todayPrice = 20;

   public Price(int price) {        todayPrice = price - staticPrice;
   }
   public static void main(String[] args) {        System.out.println(Price.INSTANCE.todayPrice);    }
   @BeforeEach    public void init(){        ArrayList<String> list = new ArrayList<String>();        System.out.println(                "Set up for "+ list.get(1)        );        for (int i = 0; i < 3; i++) {            list.add(Integer.toString(i));
       }    }    @AfterEach    public void clean(){        System.out.println(                "Clean ..."        );
   }
   @Test     public void  replace(){         System.out.println(                 "Runing testReplace()"         );
    }
}

如何解析JVM 字節(jié)碼指令

加載和存儲(chǔ)指令

        加載和存儲(chǔ)指令用于數(shù)據(jù)在棧幀中的局部變量表和操作數(shù)棧之間的來回傳輸。

        將一個(gè)局部變量加載到操作數(shù)棧:iload、iload_、lload、lload_、fload、fload_、dload、dload、aload、aload。

        將一個(gè)數(shù)值從操作數(shù)棧存儲(chǔ)到局部變量表:istore、istore_、lstore、lstore_、fstore、fstore_、dstore、dstore_、astore、astore_。

        將一個(gè)常量加載到操作數(shù)棧:bipush、sipush、ldc、ldc_w、ldc2_w、aconst_null、iconst_ml、iconst_、lconst_、fconst_、dconst_。

        擴(kuò)充局部變量表的訪問索引的指令:wide。

對(duì)象創(chuàng)建與訪問指令

        對(duì)于普通對(duì)象和數(shù)組的創(chuàng)建,JVM分別使用了不同的指令去處理。

        創(chuàng)建普通對(duì)象的指令:new

        創(chuàng)建數(shù)組的指令:newarray、anewarray、multianewarray

        訪問類變量(static類型)和實(shí)例變量(非static類型)的指令:getstatic、putstatic、getfield、putfield

        把一個(gè)數(shù)組加載到操作數(shù)棧的指令:baload、caload、saload、iaload、laload、faload、daload、aaload

        將一個(gè)操作數(shù)棧的值存儲(chǔ)到數(shù)組元素中的指令:bastore、castore、sastore、iastore、fastore、dastore、aastore

        取數(shù)組長(zhǎng)度的指令:arraylength

        檢查普通對(duì)象類型的指令:instanceof、checkcast

如何解析JVM 字節(jié)碼指令

看完上述內(nèi)容,你們掌握如何解析JVM 字節(jié)碼指令的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

jvm
AI