JVM notes
How java code run?
1 | $ echo ' |
Check the change:
1 | $ diff Foo.jasm Foo.jasm.1 |
Java basic types
Class Loader
load
tips: In this time,JVM has generated the class structure and write in the method area
Every class loader inherit java.lang.ClassLoader except boot class loader.
- Boot: jre/lib, Xbootclasspath
- Extension: ext, dirs
- Application:
JAVA9: change extension to platform which would load almost all modules except important moudles such as java.base.
Class unique ID: result of class loader object and class full path name.
link
tips: get the address of method instead of class.
Valid byte for jvm criterion.
Prepare
- Allocate memory
- Construct layer about class such as method table which realize dynamic binding of virtual method.
analyze
- resolve these symbols to actual refference
Besides: Java machine specification not require finish the analyze in the link process, it just has provisions that the necessity of resolving symbol refference before execute the corresponding byte code.
init process
final static [basic type or String]
will be signed ConstantValue and init directly by jvm. Besides, all of directly assignment and static block code will be put in the same method and rename as<clinit>
.JVM will use block method to ensure teh clinit method only be executed once.(And the feature always be used to realized singleton lazy init process)
When will the init process be triggered:
- JVM start, init user settle Main class
new
the aim class- instructions: call static method, init over class
- instructions: visit static field init over class
- child class init trigger parent class init
- default method defined in interface will result in interface init process.
- reflection init class
- the first call MethodHandle object, init the class where methodHandle locate.
1 | public class Singleton { |
The above code:
Only class Singleton.getInstance, program will visit LazyHolder.INSTANCE, then trigger the process of LazyHolder init and create a Singleton object.
Practice:
JVM: -verbose:class
will print the order of class loading and special symbol before LazyHolder method.
1 |
|
Problem:
new array whether result in LazyHolder load or init?
A: load not init.
1 | $ java -cp /path/to/asmtools.jar org.openjdk.asmtools.jdis.Main Singleton\$LazyHolder.class > Singleton\$LazyHolder.jasm.1 |
Problem2:
new array will result in link of LazyHolder?
A: No.(while getInstance(false) will link and init)
How do JVM call method?
1 | void invoke(Object obj, Object... args) { ... } |
jvm 实战
本文内容笔记主要来自
- mashibing: http://www.mashibing.com/
- 以及深入理解JVM虚拟机
- 以及深入拆解Java虚拟机(郑雨迪)极客时间 https://time.geekbang.org/column/article/41245
编译
在 HotSpot 里面,上述翻译过程有两种形式:
第一种是解释执行,即逐条将字节码翻译成机器码并执行;
第二种是即时编译(Just-In-Time compilation,JIT),即将一个方法中包含的所有字节码编译成机器码后再执行。
Java 字节码
基本类型
在 Java 中,正无穷和负无穷是有确切的值,在内存中分别等同于十六进制整数 0x7F800000 和 0xFF800000。
[0x7F800001, 0x7FFFFFFF]和[0xFF800001, 0xFFFFFFFF] 对应的浮点数是 NaN(Not-a-Number)
1 |
|
类加载 查找字节流的过程
双亲委派模型
三个GC算法
mark sweep: 标记清除,标记后进行清除,所以碎片化
copy: 标记后统一整理放到一侧,然后对另一侧大扫除。由于 Eden 的特点是大量产生,大量销毁,因此很适用copy。使用时只需将Eden 少量存活的copy到 survivor1 中,然后对 Eden
区进行清理即可。 等到二次回收时会将 eden 区的以及 survivor1 的一起复制到 survivor2 当中去。
- mark compact: 一边清一边压缩,时刻保持内存紧致。对于老年代这种对象一般都存活很久的就很适用。
10 个 Garbage collector
- Epsilon: 跟进追踪,不做任何回收,常用来 debug 或者不需要回收的场景
分代模型
基本命令
- top 查内存占用
jps 列出系统中java 的进程
jinfo pid 拿到基本信息
jstack pid | more 罗列出所有线程的调用栈状态
上面仨命令常用来定位 死锁
jstat -gc 不好用,效果不直观
arthas 基本命令:
- dashboard 类似 top,能给出基本占用信息:如果是JVM没有报OOM便退出了,建议排查堆外内存
- jvm 类似 jinfo 输出使用 JVM的相关信息比如使用的GC,memory 等
- thread 类似 jstack,加 -b 参数可用于排查死锁
- redefine 重定义 class 类似字节码插桩
jmap -histo pid 定位哪些对象在吃内存
- jmap -dump:format=b.file=xxx.hprof 2366
OQL 学习
- Post title:JVM notes
- Post author:ReZero
- Create time:2020-09-18 12:33:00
- Post link:https://rezeros.github.io/2020/09/18/jvm/
- Copyright Notice:All articles in this blog are licensed under BY-NC-SA unless stating additionally.