JVM调优
堆(Heap)和非堆(Non-heap)内存
按照官方的说法:“Java 虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。堆是在 Java 虚拟机启动时创建的。”“在JVM中堆之外的内存称为非堆内存(Non-heap memory)”。可以看出JVM主要管理两种类型的内存:堆和非堆。简单来说堆就是Java代码可及的内存,是留给开发人员使用的;非堆就是JVM留给自己用的,所以方法区、JVM内部处理或优化所需的内存(如JIT编译后的代码缓存)、每个类结构(如运行时常数池、字段和方法数据)以及方法和构造方法的代码都在非堆内存中。
堆内存分配
-Xmx:最大JVM可用内存
-Xms:最小JVM可用内存
JVM初始分配的内存由-Xms指定,默认是物理内存的1/64;JVM最大分配的内存由-Xmx指定,默认是物理内存的1/4。默认空余堆内存小于40%时,JVM就会增大堆直到-Xmx的最大限制;空余堆内存大于70%时,JVM会减少堆直到-Xms的最小限制。因此服务器一般设置-Xms、-Xmx相等以避免在每次GC 后调整堆的大小。
非堆内存分配
JVM使用-XX:PermSize设置非堆内存初始值,默认是物理内存的1/64;由XX:MaxPermSize设置最大非堆内存的大小,默认是物理内存的1/4。
//缩小堆内存的时机-XX:MaxHeapFreeRatio=70堆内存使用率大于70时扩张堆内存,如果最大堆内存=初始堆内存时该参数无效,默认值70
//扩张堆内存的时机-XX:MinHeapFreeRatio=40堆内存使用率小于40时缩减堆内存,如果最大堆内存=初始堆内存时该参数无效,默认值40
提高Java执行效率
1)什么是Hotspot JIT编译器?
你的应用程序可能有数百万行的代码。然而,只有一小部分代码会被反复执行。这个小的代码子集(也被称为 “热点”)负责你的应用程序的性能。在运行时,JVM使用这个JIT(Just in time)编译器来优化这个热点代码。大多数时候,由应用程序开发人员编写的代码并不是最佳的。因此,JVM的JIT编译器对开发者的代码进行优化,以提高性能。为了进行这种优化,JIT编译器使用C1、C2编译器线程。
2)什么是代码缓存?
JIT编译器用于代码编译的内存区域被称为 “代码缓存”。这个区域位于JVM堆和元空间之外。要了解不同的JVM内存区域,你可以参考这个视频片段。
3)c1和c2编译器线程之间的区别是什么?
在Java的早期,有两种类型的JIT编译器。
a. 客户端
b. 服务器
根据你想使用的JIT编译器的类型,必须下载和安装适当的JDK。例如,如果你正在构建一个桌面应用程序,那么需要下载具有 "客户端 "JIT编译器的JDK。如果你要构建一个服务器应用程序,那么就需要下载具有 "服务器 "JIT编译器的JDK。
客户端 JIT 编译器会在应用程序启动后立即开始编译代码。服务器JIT编译器将观察代码的执行情况相当一段时间。基于它获得的执行知识,它将开始进行JIT编译。尽管服务器JIT编译速度很慢,但它产生的代码将比客户端JIT编译器产生的代码更优秀、更有性能。
今天,现代的JDK同时带有客户端和服务器JIT编译器。这两种编译器都试图优化应用程序的代码。在应用程序启动时,代码是用客户端JIT编译器编译的。后来,随着知识的增加,代码就用服务器JIT编译器进行编译。这被称为JVM中的分层编译。
JDK的开发者称他们为客户端和服务器JIT编译器,内部称为c1和c2编译器。因此,客户端JIT编译器使用的线程被称为c1编译器线程。服务器JIT编译器使用的线程被称为c2编译器线程。
-XX:TieredStopAtLevel=N 有四个层次的编译:
Compilation Level Description
0 Interpreted Code
1 Simple c1 compiled code
2 Limited c1 compiled code
3 Full c1 compiled code
4 C2 compiled code
| Compilation Level | Description |
|---|---|
| 0 | Interpreted Code |
| 1 | Simple c1 compiled code |
| 2 | Limited c1 compiled code |
| 3 | Full c1 compiled code |
| 4 | C2 compiled code |

3021

被折叠的 条评论
为什么被折叠?



