并发王者课-青铜6:借花献佛-如何格式化Java内存工具JOL输出
2021-05-27 10:29
633 查看
在前面的文章《一探究竟-如何从synchronized理解Java对象头中的锁》中,我们介绍并体验了JOL工具。虽然JOL很赞,但它的输出对我们不是很友好,如果不借助工具,我们很难直观理解其中的含义。
下面这段代码是对JOL输出的翻译,建议你收藏。代码非我原创,文末已经注明出处。
import org.openjdk.jol.info.ClassLayout; import java.nio.ByteOrder; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.regex.Matcher; import java.util.regex.Pattern; public class PrintObjectHeader { /** * Get binary data * * @param o * @return */ public static String getObjectHeader(Object o) { ByteOrder order = ByteOrder.nativeOrder();//Byte order String table = ClassLayout.parseInstance(o).toPrintable(); Pattern p = Pattern.compile("(0|1){8}"); Matcher matcher = p.matcher(table); List<String> header = new ArrayList<String>(); while (matcher.find()) { header.add(matcher.group()); } //Little-endian machines, need to traverse in reverse StringBuilder sb = new StringBuilder(); if (order.equals(ByteOrder.LITTLE_ENDIAN)) { Collections.reverse(header); } for (String s : header) { sb.append(s).append(" "); } return sb.toString().trim(); } /** * Parsing object header function for 64bit jvm * In 64bit jvm, the object header has two parts: Mark Word and Class Pointer, Mark Word takes 8 bytes, Class Pointer takes 4 bytes * * @param s Binary string of object header (each 8 bits, separated by a space) */ public static void parseObjectHeader(String s) { String[] tmp = s.split(" "); System.out.print("Class Pointer: "); for (int i = 0; i < 4; ++i) { System.out.print(tmp[i] + " "); } System.out.println("\nMark Word:"); if (tmp[11].charAt(5) == '0' && tmp[11].substring(6).equals("01")) {//0 01 lock-free state, regardless of GC mark //notice: Mark word structure without lock: unused(25bit) + hashcode(31bit) + unused(1bit) + age(4bit) + biased_lock_flag(1bit) + lock_type(2bit) // The reason why hashcode only needs 31bit is: hashcode can only be greater than or equal to 0, eliminating the negative range, so you can use 31bit to store System.out.print("\thashcode (31bit): "); System.out.print(tmp[7].substring(1) + " "); for (int i = 8; i < 11; ++i) System.out.print(tmp[i] + " "); System.out.println(); } else if (tmp[11].charAt(5) == '1' && tmp[11].substring(6).equals("01")) {//1 01, which is the case of biased lock //notice: The object is in a biased lock, its structure is: ThreadID(54bit) + epoch(2bit) + unused(1bit) + age(4bit) + biased_lock_flag(1bit) + lock_type(2bit) // ThreadID here is the thread ID holding the biased lock, epoch: a timestamp of the biased lock, used for optimization of the biased lock System.out.print("\tThreadID(54bit): "); for (int i = 4; i < 10; ++i) System.out.print(tmp[i] + " "); System.out.println(tmp[10].substring(0, 6)); System.out.println("\tepoch: " + tmp[10].substring(6)); } else {//In the case of lightweight locks or heavyweight locks, regardless of the GC mark //notice: JavaThread*(62bit,include zero padding) + lock_type(2bit) // At this point, JavaThread* points to the monitor of the lock record/heavyweight lock in the stack System.out.print("\tjavaThread*(62bit,include zero padding): "); for (int i = 4; i < 11; ++i) System.out.print(tmp[i] + " "); System.out.println(tmp[11].substring(0, 6)); System.out.println("\tLockFlag (2bit): " + tmp[11].substring(6)); System.out.println(); return; } System.out.println("\tage (4bit): " + tmp[11].substring(1, 5)); System.out.println("\tbiasedLockFlag (1bit): " + tmp[11].charAt(5)); System.out.println("\tLockFlag (2bit): " + tmp[11].substring(6)); System.out.println(); } public static void printObjectHeader(Object o) { if (o == null) { System.out.println("null object."); return; } parseObjectHeader(getObjectHeader(o)); } }
参考资料
关于作者
关注公众号【庸人技术笑谈】,获取及时文章更新。记录平凡人的技术故事,分享有品质(尽量)的技术文章,偶尔也聊聊生活和理想。不贩卖焦虑,不兜售课程。
如果本文对你有帮助,欢迎点赞、关注。
相关文章推荐
- java如何将json数据格式化输出到控制台
- JAVA内存泄露检测工具如何使用
- 如何让Java应用把当前内存情况输出到文件中供分析
- 如何排查Java内存泄露(内附各种排查工具介绍)
- Java内存操作流、格式化输出、打印流
- JAVA中使用printf(System.out.printf)如何格式化输出数字
- 并发王者课-青铜7:顺藤摸瓜-如何从synchronized中的锁认识Monitor
- java字符串与格式化如何输出
- 并发王者课-青铜10:千锤百炼-如何解决生产者与消费者经典问题
- JAVA中如何输出一个特殊符号?
- java 格式化输出
- [转]如何在Eclipse 3.1.1上安装jadclipse[java的反编译工具]
- java实战技巧--关于格式化输出日期
- Java格式化输出
- java中格式化输出数字
- java应用中如何捕抓SAS存储过程输出的流信息
- 如何正确使用Java I/O输出和读入数据
- 如何获取Java中内存对象的大小? sizeof = ?
- Java中日期Date的格式化输出
- 如何得到Java应用程序的可用内存?