jvm1

java JVM内存模型
%.1.java监控工具使用
%.2.Jps查看jvm的进程
%.2.1.jconsole 
    jconsole是一种集成了上面所有命令功能的可视化工具,可以分析jvm的内存使用情况和线程等信息。

通过JDK/bin目录下的“jconsole.exe”启动Jconsole后,将自动搜索出本机运行的所有JVM进程,不需要用户使用jps来查询了,双击其中一个进程即可开始监控。也可以“远程连接服务器,进行远程虚拟机的监控。”

Main进程是本地的一个springboot项目。

远程连接:jvisualvmjconsole是一样的。

概述页面显示的是整个虚拟机主要运行数据的概览。非堆区是方法区(永久区)。

.1.1. jvisualvmjconsole更好用,也在bin里面

提供了和jconsole的功能类似,提供了一大堆的插件。

插件中,Visual GC(可视化GC)还是比较好用的,可视化GC可以看到内存的具体使用情况。

本地进程(jvm进程):eclipseJConsoleVisualVM(是他自己),

Dump出来可以有一个很详细的线程报告:

Eclipsetomcat也有性能监控工具。

 

%.1.java内存模型
%.1.1.内存模型图解
    虚拟机把所管理的内存区域划分为若干个不同的区。这些区域有各自的用途以及创建和销毁的时间。
有的区域随着虚拟机进程的启动而存在,有的区域则依赖用户线程的启动和结束而建立和销毁,我们可以将这些区域统称为Java运行时数据区域。

Java虚拟机运行时数据区域被分为五个区域:堆(Heap)、栈(Stack)、本地方法栈(Native Stack)、方法区(Method Area)、程序计数器(Program Count Register)。

    类的定义放在一个地方,放在方法区里面Method Area也叫永久区。
程序运行会产生一个个的对象,放在堆里面。
栈(Stack)是程序执行的时候本质是启动一个main线程执行,main线程执行方法的时候有局部变量放在栈里面。
栈里面是分线程存放的,每个线程都有自己的栈空间,保证每个线程不冲突。堆里面是线程共享的。一个线程会调用多个嵌套的方法a调B,B调C,比如递归里面。线程每调用一个方法就会产生一个栈桢,存放方法调用过程中需要的变量。调用的方法过多就会产生栈溢出。

 

程序计数器(Program Count Register):执行到哪一行了。

本地方法栈(Native Stack):jvm跟操作系统的交互,调用windowslinux的方法,比如文件操作都会调用本地操作系统的方法。

垃圾回收:C++里面maloc之后要free掉,

%.1.1. 堆(Heap)
    Java Heap区域随着虚拟机的启动而创建。是一块共享的区域有锁和同步。
堆分为新生代、老年代、永久区(方法区)。垃圾回收也要回收永久区(静态变量,类的定义),永久区也会溢出,所以垃圾回收也回收永久区。
新生代、老年代根据对象存活的年龄划分的。New的新的对象在新生代,垃圾回收器

垃圾回收器从程序的根节点开始搜索,看能够走到哪些对象,A引用B,B引用A,但是程序到达不了这个对象,虽然A和B都有引用(根据引用判断垃圾是不准确的),但是垃圾回收器到达不了,还是会把他当成垃圾回收。

收集器的使用(配置文件配置):

现在大多数的GC基本都采用了分代收集算法。如果再细致一点,Java Heap还有Eden空间,From Survivor空间,To Survivor空间等。

Java Heap可以处于物理上不连续的内存空间中,只要逻辑上是连续的即可。

.1.1. 

.1.1.  栈(Stack

相对于Java Heap来讲,Java Stack是线程私有的,她的生命周期与线程相同。Java Stack描述的是Java方法执行时的内存模型,每个方法执行时都会创建一个栈帧Stack Frame)用语存储局部变量表、操作数栈、动态链接、方法出口等信息。从下图从可以看到,每个线程在执行一个方法时,都意味着有一个栈帧在当前线程对应的栈帧中入栈和出栈。

图中可以看到每一个栈帧中都有局部变量表。局部变量表存放了编译期间的各种基本数据类型,对象引用等信息。

%.1.1.本地方法栈(Native Stack)
    本地方法栈(Native Stack)与Java虚拟机站(Java Stack)所发挥的作用非常相似,他们之间的区别在于虚拟机栈为虚拟机栈执行java方法(也就是字节码)服务,而本地方法栈则为使用到Native方法服务。与操作系统的方法交互。
%.1.2.方法区(Method Area)
    方法区(Method Area)与堆(Java Heap)一样,是各个线程共享的内存区域,它用于存储虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据(会把jsp编译成servlet的java代码)。虽然Java虚拟机规范把方法区描述为堆的一个逻辑部分,但是她却有一个别名叫做非堆(Non-Heap)。分析下Java虚拟机规范,之所以把方法区描述为堆的一个逻辑部分,应该觉得她们都是存储数据的角度出发的。一个存储对象数据(堆),一个存储静态信息(方法区)。
    在上文中,我们看到堆中有新生代、老生代、永久代的描述。为什么我们将新生代、老生代、永久代三个概念一起说,那是因为HotSpot虚拟机的设计团队选择把GC分代收集扩展至方法区,或者说使用永久代来实现方法区而已。这样HotSpot的垃圾收集器就能想管理Java堆一样管理这部分内存。简单点说就是HotSpot虚拟机中内存模型的分代,其中新生代和老生代在堆中,永久代使用方法区实现。根据官方发布的路线图信息,现在也有放弃永久代并逐步采用Native Memory来实现方法区的规划,在JDK1.7的HotSpot中,已经把原本放在永久代的字符串常量池移出。
%.1.1.总结
1、线程私有的数据区域有:
 Java虚拟机栈(Java Stack)
本地方法栈(Native Stack)
2、线程共有的数据区域有:
堆(Java Heap)
方法区

.1. JVM参数列表

java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:NewRatio=4 -XX:SurvivorRatio=4 -XX:MaxPermSize=16m  -XX:MaxTenuringThreshold=0
-Xmx3550m:最大堆内存为3550M。
-Xms3550m:初始堆内存为3550m。
此值可以设置与-Xmx相同,以避免每次垃圾回收完成后JVM重新分配内存。
-Xmn2g:设置年轻代大小为2G。
整个堆大小=年轻代大小 + 年老代大小 + 持久代大小。持久代一般固定大小为64m,所以增大年轻代后,将会减小年老代大小。此值对系统性能影响较大,Sun官方推荐配置为整个堆的3/8-Xss128k:设置每个线程的栈大小。
JDK5.0以后每个线程栈大小为1M,在相同物理内存下,减小这个值能生成更多的线程。但是操作系统对一个进程内的线程数还是有限制的,不能无限生成,经验值在 3000~5000左右。 
-XX:NewRatio=4:设置年轻代(包括Eden和两个Survivor区)与年老代的比值(除去持久代)。设置为4,则年轻代与年老代所占比值为1:4,年轻代占整个堆栈的1/5
-XX:SurvivorRatio=4:设置年轻代中Eden区与Survivor区的大小比值。
设置为4,则两个Survivor区与一个Eden区的比值为2:4,一个Survivor区占整个年轻代的1/6
-XX:MaxPermSize=16m:设置持久代大小为16m。
-XX:MaxTenuringThreshold=0:设置垃圾最大年龄。
如果设置为0的话,则年轻代对象不经过Survivor区,直 接进入年老代。对于年老代比较多的应用,可以提高效率。如果将此值设置为一个较大值,则年轻代对象会在Survivor区进行多次复制,这样可以增加对象 再年轻代的存活时间,增加在年轻代即被回收的概论。


收集器设置
-XX:+UseSerialGC:设置串行收集器
-XX:+UseParallelGC:设置并行收集器
-XX:+UseParalledlOldGC:设置并行年老代收集器
-XX:+UseConcMarkSweepGC:设置并发收集器
垃圾回收统计信息
-XX:+PrintGC :jvm故障调试的时候
-XX:+PrintGCDetails
-XX:+PrintGCTimeStamps
-Xloggc:filename
并行收集器设置
-XX:ParallelGCThreads=n:设置并行收集器收集时使用的CPU数。并行收集线程数。
-XX:MaxGCPauseMillis=n:设置并行收集最大暂停时间
-XX:GCTimeRatio=n:设置垃圾回收时间占程序运行时间的百分比。公式为1/(1+n)
并发收集器设置
-XX:+CMSIncrementalMode:设置为增量模式。适用于单CPU情况。
-XX:ParallelGCThreads=n:设置并发收集器年轻代收集方式为并行收集时,使用的CPU数。并行收集线程数。

 

posted @ 2018-05-26 21:51  无天666  阅读(264)  评论(0编辑  收藏  举报