快速掌握JVM,小弟5篇总结的博客

http://blog.csdn.net/qq_25235807/article/details/61920877#Java工程师#
全部评论
辛苦***帮忙粘帖过来吧:)
点赞 回复 分享
发布于 2017-03-14 20:48
java程序的运行流程 众所周知所有的Java程序代码必须保存在.java的文件之中,这些称为源代码。而这些源代码并不能够直接执行,必须使用javac.exe命令将其编译为.class文件,而后利用java.exe命令在JVM进程之中解释此程序。 实际上当JVM将所需要的*.class文件加载到JVM进程之中,那么这个过程就需要有一个类加载器(ClassLoader),有了类加载器的好处在于:可以随意指定程序 .class文件的所在路径。 类加载器概念 JVM:Java虚拟机,所有的程序都要求运行在JVM上,是因为考虑到了可移植性问题,可是如果要想真正去执行程序,你绝对不可能离开操作系统的支持。 在Java里面可以使用native实现本地C函数的调用。但是这些都属于程序运行的辅助手段,而真正的程序运行都在“运行时数据区”之中。 运行时数据区 运行时数据区由以下几个部分组成: 堆内存:保存所有引用数据类型的真实信息 栈内存:基本类型,运算,指向堆内存的指针 方法区:所有定义方法信息,属于共享区 程序计数器:是一个很小的内存空间,用于执行下个执行的代码 本地方法栈:每一次执行递归的方法处理都会将上一个方法入栈 栈 栈内存是线程私有的,栈的生命周期和线程相同;栈描述的是java方法执行的内存模型,执行一个方法的时候会产生一个栈帧,随后将其保存到栈顶,方法执行完毕自动出栈。顶部的栈帧表示的是当前的方法;注意的是如果栈的深度过大。如递归过深虚拟机会抛出StackOverError。如果虚拟机允许栈动态扩展内存不足时抛出OutOfMemoryError. 栈帧 局部变量表:方法的局部变量或形参 操作数栈:表达式的计算 方法所属类的运行时常量池常量池的引用:引用其他类常量或者字 符 串常量池的字符串 方法返回地址:方法执行完后需要返回调用该方法的位置,所以要在栈帧中保存方法的返回地址 关于常量池的简单介绍 在整个Java之中存在有对象池的概念,对象池是对整个常量池的一个规则破坏,因为在JVM启动的时候所有的常量都已经分配好内存空间了,但是String中的intern()方法却可以打破这种限制,动态的进行常量池的内容设置 对象的访问模式 Obiect obj=new Object();为例 Obiect obj 保存在栈内存中,有对应的堆内存引用 new Object() 一个真的对象保存在堆内存中 引用操作 新定义的的对象名称保存在本地变量表中,而后在这块区域确定与之对应的栈空间通过栈地址找到堆内存,利用堆内存对象调用本地方法。 实际上所有的引用数据类型的访问有两种模式:一是通过句柄访问,一是通过对象指针直接访问。但是在Java之中它直接利用的是对象保存模式,也就是说堆内存里面不再需要保存句柄,而直接保存具体的对象。就相当于省略了句柄到对象间的查找。而后这个对象可以直接进行Java方法区的调用。 通过直接指针访问–HotSpot虚拟机(jvm) 引用数据类型在HotSpot中都是直接进行的引用处理,没有句柄池的概念。
点赞 回复 分享
发布于 2017-03-14 21:40
Java具备良好的垃圾收集特性,也就是GC是整个Java之中的最重要的安全保证,它可以保证即使再蠢的开发者也能够写合理的代码来。整个JVM中的GC的处理机制:对不需要的对象进行标记,而后进行清除. java堆内存模型 JDK1.7以前 JDK1.8 JDK 1.8之后将最初的永久代内存空间取消了为了将HotSpot与JRockit两个虚拟机标准联合为一个. 新生代:新的对象和没有达到一定“年龄”的对象对象存放空间(活跃对象) 老年代:被长时间使用的对象,老年代空间要相对较大 元空间:一些操作的临时对象,如方法中的临时对象,直接使用物理内存 注意:永久代和元空间都是存放临时对象的,但是永久代使用的是JVM直接分配的内存,而元空间使用的是物理内存 GC的流程(非常重要) java所有的数据都会保存在JVM的堆内存之中,但是在实际的开发之中经常会创建许多的临时对象,也会有一些常驻对象存在,所以为了保证GC的性能,就要进行垃圾回收。 GC的处理流程 注意:在整个GC的流程之中是 针对新生代和老年代进行内存清理的操作,而元空间和永久代都不在GC的范围之内。 重要: 1. 当新创建一个对象的时候,那么对象一定需要在堆内存中分配内存空间。所以就需要对该对象申请内存空间 2. 首先会判断Eden区中是否有充足的内存空间,如果有,那么直接将对象保存在Eden区中 3. 如果此时Eden中内存空间不足,那么会自动执行一个Minor GC的操作,将伊甸园区的无用的内存空间进行清理,清理之后 继续判断伊甸园区的内存空间是否充足。如果充足,那么将对象直接在疑点园区中进行内存分配。 4. 如果执行了MinorGC之后发现伊甸园区的内存依然不足,则会对存活区进行判断,如果存活区内存足够。那么将伊甸园区的 一部分活跃对象保存到存活区,随后继续判断伊甸园区内存,如果够进行新对象的内存分配。 5. 如果此时存活区没有足够的内存空间,则继续判断老年区。如果老年区的内存空间充足,则将存活的部分活跃对象保存到 老年代,而后存活区会出现剩余空间。随后将伊甸园区的活跃对象保存对象, 然后在伊甸园区开辟空间,保存新对象。 6. 如果此时老年代也没有剩余空间,则执行MajorGC(FullGC),清理老年代内存 7. 如果执行了FullGC之后依然无法保存对象,就会产生OOM异常“OutofMemoryError”。 解释的是:StackOverError是由于栈的深度过大由和栈帧有关的异常,请看第一篇。而OutofMemoryError是由于堆内存引起的。
点赞 回复 分享
发布于 2017-03-14 21:41
堆内存调整参数(调优的关键) 通过以上两篇的文章我们可以发现每一个区域都有一个可变的伸缩区,当我们的内存空间不足的时候,会在可变的范围内扩大内存空间,当我们的内存空间变得不紧张的时候我们再释放可变空间。 在堆内存的调优之中我们要特别注意两个参数-Xms初始化内存分配大小,默认为物理内存的1/64,-Xmx 最大的分配内存默认为物理内存的1/4。 以本人8G电脑来说: class GCDemo { public static void main(String[] args) { Runtime run =Runtime.getRuntime(); long maxMemory =run.maxMemory(); long totalMemory =run.totalMemory(); System.out.println("总内存"+(maxMemory/(double)1024/1024)+"M"); System.out.println("可用内存"+(totalMemory/(double)1024/1024)+"M"); } } 1 2 3 4 5 6 7 8 9 10 11 1 2 3 4 5 6 7 8 9 10 11 输出GC的处理日志: 那么现在的可调整内存空间(伸缩区)就在123M-1810M之间,这样可能造成程序的性能下降,所以我们将初始内存空间和最大的内存空间设置为相等。就避免了伸缩区的可调策略。从而提升程序的性能。 输出GC的处理日志: 可以看出调整后的堆内存的每个区域都比以前大了,此时没有可伸缩的区域,而且也解释了前面的问题,老年代的内存空间是最大的,而两个存活的内存一定是相等的。 年轻代参数调整 所有使用关键字new新实例化的对象一定会在伊甸园区进行保存,而对于存活区保存的一定是已经在伊甸园区中存在好久,并且经过了好几次的Minor GC还保存下来的活跃对象。那么这个对象将晋升到存活区之中,存活区一定会有两块空间,这两块空间的大小一定是相等的,目的:一块存活区为了晋升、另外一块存活区为了对象回收。这两块内存空间一定有一块是空的。在年轻代中使用的是Minor GC,这种GC算法采用的是复制算法。 Minor GC 实现算法: 首先扫面伊甸园区中的存活对象,其中有活跃对象和不活跃对象。此时会将活跃的对象晋升到其两个存活区中的那个空的存活去中而另一个存活区中的活跃对象也会复制到这个存活区,如果该存活区还有不活跃对象会晋升到老年代中。以保证必须有一个空的存活区这时伊甸园会产生新的空间,保存新的对象。 通过以上分析会发现伊甸园区中保存的对象大部分可能是临时对象,那么这就会频繁的发生Minor GC,所以HotSpot为了加快此空间的分配操作,采用了两种技术。 Bump-The-Pointer 该技术采用栈的形式,每一个最后保存的对象一般都会保存在伊甸园区的顶部,这样的话只要每创建一个新的对象我们就只需要检查保存对象后面是否有新的空间即可。 TLAB Bump-The-Pointer虽然提高了内存的分配速度但是如果是多个线程同时保存一个对象那?显然是不适合的所以TLAB算法又将伊甸园区分为多个数据块,每个数据块使用Bump-The-Pointer进行对象的保存和分配。 老年代参数调整 老年代主要是接收由年轻代发送来的对象,一般情况下经过了好几次的Minor GC之后还会保存下来的对象才会进入到老年代。如果你要保存的对象超过了伊甸园区的大小,那么此对象也将直接保存到老年代之中。当老年代内存不足时将引发“Full GC在老年代里面会采用两种算法结合的模式实现GC的处理:整理-压缩。 标记-清除 在活跃对象较多时比较高效,但是由于该算法直接回收内存对象会造成内存的碎片化这也是最头疼的问题。 标记-压缩 在标记-清除的基础上进行对象的移动,成本较高,但是不产生碎片。类似于电脑的磁盘管理。 永久代参数调整 虽然jdk1.8中没有永久代但是javaEE目前依然是1.7 元空间参数调整 总结 1、JDK 1.8中的内存分为:年轻代、老年代、元空间; 2、年轻代有两个组成部分:伊甸园区、两个存活区,GC回收策略为“复制”; 3、老年代的保存空间一般较大,使用的是“整理-压缩”GC回收策略。 4、JDK 1.8之前存在有永久代; 面试题: 一般什么时候会发生GC?如何处理? Java中的GC会有两种回收:年轻代的Minor GC,另外一个就是老年代的Full GC;新对象创建时如果伊甸园空间不足会触发 MinorGC,如果此时老年代的内存空间不足会触发Full GC,如果空间都不足抛出OutOfM emoryError 。 面试题: 如果整个Java项目运行缓慢,你该如何解决? 避免堆内存的伸缩空间进行操作,将初始化内存空间与整体堆内存空间大小设置为一一样,使用“-Xms”、“-Xmx” 面试题: 当一个对象非常大的时候,请问是如何保存的? 该对象将直接保存在老年代。
点赞 回复 分享
发布于 2017-03-14 21:41

相关推荐

牛客963010790号:为什么还要收藏
点赞 评论 收藏
分享
点赞 评论 收藏
分享
点赞 收藏 评论
分享
牛客网
牛客企业服务