JDK体系结构
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。 *Error 示例: 1 // JVM 设置 ‐Xss128k( 默认 1M) 2 public class *Test { 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 . *Error 23 at com . tuling . jvm . *Test . redo ( *Test . java : 12 ) 24 at com . tuling . jvm . *Test . redo ( *Test . java : 13 ) 25 at com . tuling . jvm . *Test . redo ( *Test . java : 13 ) 26 ...... 结论: -Xss设置越小count值越小,说明一个线程栈里能分配的栈帧就越少,但是对JVM整体来说能开启的线程数会更多