JDK体系结构
![](https://i-blog.csdnimg.cn/blog_migrate/815afb4435738174680d227f562ac588.png)
Java语言的跨平台特性
JVM整体结构及内存模型
堆:公共的内存区域,存储对象
方法区:公共内存区域,存储常量池,静态变量,类信息 如 final String a=new String("");
会指向堆
栈:线程独自持有的区域,存储局部变量,执行代码时会在内存中给每个线程跑的程序分配一块栈区域,在栈中每次调用一个方法会分配一个栈帧区域,如main方法中调用compute方法 即会在栈中分配两个栈帧,在每个栈帧中会有局部变量表,操作数栈,动态链接,方法出口
如 int a=0; int b=1;
先会在局部变量表定义a 操作数栈中定义 1 ,之后1从操作数栈出栈 局部变量表 a变为 a=1
方法出口即是让方法知道出口在哪里 指向对应的地方
本地方法栈:即调用底层c语言的方法,一般用native修饰
程序计数器:来控制程序执行的顺序走向,如果多线程执行线程唤醒之后需要知道下一步执行哪行代码
堆中的gc
首先每次new的对象都会放入堆中的伊甸区
伊甸区满了之后会发生一次minor gc
会追踪整条链路的对象关系把他们取出来,之后把还存活的对象放入到s0中,打一个标识为1,失效的对象就直接舍弃了
等再次发生minor gc 之后 会把存活的对象放入到s1,做一个交换,这时候标识会+1
直到幸存者区s0 s1发生了15次交换,标识到达15时或者满了的时候,会放入老年代
如果老年代区域的内存空间被放满了,会发生full jc
minor jc 和full jc会停掉用户进程,即发生stw stop the world
为什么要设计stw?
如果不停止用户进程 在jc时会寻找链路对象,但是这时候程序还在执行,可能已经执行完了,变成了垃圾对象,这时候会认为他是一个有效的对象,即不进行回收操作
JVM内存参数设置
Spring Boot程序的JVM参数设置格式(Tomcat启动直接加在bin目录下catalina.sh文件里):
1
java
‐
Xms2048M
‐
Xmx2048M
‐
Xmn1024M
‐
Xss512K
‐
XX
:
MetaspaceSize
=
256
M
‐
XX
:
MaxMetaspaceSize
=
256
M
‐
jar microservice
‐
eurek
a
‐
server
.
jar
关于元空间的JVM参数有两个:-XX:MetaspaceSize=N和 -XX:MaxMetaspaceSize=N
-XX:MaxMetaspaceSize
: 设置元空间最大值, 默认是-1, 即不限制, 或者说只受限于本地内存大小。
-XX:MetaspaceSize
: 指定元空间触发Fullgc的初始阈值(元空间无固定初始大小), 以字节为单位,默认是21M,达到该值就会触发
full gc进行类型卸载, 同时收集器会对该值进行调整: 如果释放了大量的空间, 就适当降低该值; 如果释放了很少的空间, 那么在不超
过-XX:MaxMetaspaceSize(如果设置了的话) 的情况下, 适当提高该值。这个跟早期jdk版本的
-XX:PermSize
参数意思不一样,-
XX:PermSize
代表永久代的初始容量。
由于调整元空间的大小需要Full GC,这是非常昂贵的操作,如果应用在启动的时候发生大量Full GC,通常都是由于永久代或元空间发生
了大小调整,基于这种情况,一般建议在JVM参数中将MetaspaceSize和MaxMetaspaceSize设置成一样的值,并设置得比初始值要大,
对于8G物理内存的机器来说,一般我会将这两个值都设置为256M。
StackOverflowError
示例:
1
// JVM
设置
‐Xss128k(
默认
1M)
2
public class
StackOverflowTest
{
3
4
static
int count
=
0
;
5
6
static void
redo
() {
7
count
++
;
8
redo
();
9
}
10
11
public static void
main
(
String
[]
args
) {
12
try
{
13
redo
();
14
}
catch
(
Throwable t
) {
15
t
.
printStackTrace
();
16
System
.
out
.
println
(
count
);
17
}
18
}
19
}
20
21
运行结果:
22
java
.
lang
.
StackOverflowError
23
at com
.
tuling
.
jvm
.
StackOverflowTest
.
redo
(
StackOverflowTest
.
java
:
12
)
24
at com
.
tuling
.
jvm
.
StackOverflowTest
.
redo
(
StackOverflowTest
.
java
:
13
)
25
at com
.
tuling
.
jvm
.
StackOverflowTest
.
redo
(
StackOverflowTest
.
java
:
13
)
26
......
结论:
-Xss设置越小count值越小,说明一个线程栈里能分配的栈帧就越少,但是对JVM整体来说能开启的线程数会更多