JVM内存模型和性能优化

JVM内存模型优点

  1. 内置基于内存的并发模型:      多线程机制
  2. 同步锁Synchronization
  3. 大量线程安全型库包支持
  4. 基于内存的并发机制,粒度灵活控制,灵活度高于数据库锁。
  5. 多核并行计算模型
  6. 基于线程的异步模型。


JVM性能的人为问题

  1. 关键原因是:没有正确处理好对象的生命周期。
  2. 需要从需求中找出存在自然边界的业务对象,将其对应落实到内存中,成为内存模型In-memory Domain Model。
  3. 有大小边界限制的内存是缓存,没有永远使用不完的内存,缓存=“有边界的”内存。
  4. 缓存是Domain Model对象缓存,不同于传统意义上数据库缓存的定义。
  5. 分布式缓存可以提高巨量数据处理计算能力。


Java内存种类

  1. Stack栈内存
    存取速度快,数据可多线程间共享。
    存在栈中的数据大小与生存期必须确定
  2. Heap堆内存
       大小动态变化,对象的生命周期不必事先告诉编译器JVM。

两种内存使用

  1. Stack栈内存
    基本数据类型,Java  指令代码,常量
    对象实例的引用 对象的方法代码
  2. Heap堆内存
       对象实例的属性数据和数组。堆内存由Java虚拟机的自动垃圾回收器来管理。


对象如何保存在内存中?

  1. 对象的属性Attribute Property

属性值作为数据,保存在数据区heap 中,包括属性的类型Classtype和对象本身的类型

  1. 方法method

方法本身是指令的操作码,保存在stack中。
   方法内部变量作为指令的操作数也是在Stack中,
    包括基本类型和其他对象的引用。

  1. 对象实例在heap 中分配好内存以后,需要在stack中保存一个4字节的heap内存地址,用来定位该对象实例在heap 中的位置,便于找到该对象实例。


静态属性和方法的特点

  1. 静态属性和方法都是保存在Stack中,
  2. Stack内存是共享的,其他线程都可以访问静态属性实际是全局变量。
  3. 静态方法在Stack,就无法访问Heap中的数据。静态方法无法访问普通对象中数据。
  1. 静态属性意味着全局变量,生命周期和JVM一致。JVM属于技术边界,静态只能用于技术边界内工具性质使用,不能用作业务。


内存管理:垃圾回收机制

  1. 每一种垃圾收集的算法(引用计数、复制、标记-清除和标记-整理等)在特定条件下都有其优点和缺点。
  2. 当有很多对象成为垃圾时,复制可以做得很好,但是复制许多生命周期长的对象时它就变得很糟(要反复复制它们)。
  3. 标记-整理适合生命周期长对象可以做得很好(只复制一次),但是不适合短生命的对象。
  4. Sun JVM 1.2 及以后版本使用的技术称为 分代垃圾收集(generational garbage collection),它结合了这两种技术以结合二者的长处。


可选用的GC类型



JVM性能优化

  1. 内存微调优化
  1. 锁争夺微调:

多线程 不变性 单写原则 Actor Disrupotor

  1. CPU使用率微调
  1. I/O 微调


内存微调优化

  1. 内存分配:

新生代 Eden和survior  旧生代内存大小分配。
   内存越大,吞吐量越大,但是需要内存整理的时间就越长,响应时间有延迟。

  1. 垃圾回收机制

垃圾回收启动整个应用都暂停,暂停时间造成响应时间有延迟。


内存微调目标

  1. 在延迟性(响应时间)和吞吐量上取得一个平衡。
  1. 内存大小影响吞吐量和延迟性。需要在内存大小和响应时间之间取得一个平衡。
  1. 垃圾回收机制是延迟的最大问题。目标尽量不启动,少启动。



内存模型

新生代Eden内存分配

  • 新生代(New Generation ):Eden + 1 Survivor。所有新创建的对象在Eden。
  • 当Eden满了,启动Stop-The-World的GC,或为minor gc,采取数次复制Copy-Collection到Survivor。
  • 经过几次收集,寿命不断延长的对象从Survivor 进入老生代,也称为进入保有Tenuring,类似普通缓存LRU算法。

survivor设计要旨

  1. 足够大到能容纳所有请求响应中涉及的对象数据。
  1. 每个survivor空间也要足够大到能够容纳活跃的请求对象和保有对象。
  1. Survivor大小决定了保有Tenuring阀值,阀值如果能大到容纳所有常住对象,那么保有迁移过程就越快。

老生代Old

  1. 老生代的gc称为major gc,就是通常说的full gc。
  2. 采用标记-整理算法。由于老年区域比较大,而且通常对象生命周期都比较长,标记-整理需要一定时间。所以这部分的gc时间比较长。
  3. minor gc可能引发full gc。当eden+from space的空间大于老生代的剩余空间时,会引发full gc。这是悲观算法,要确保eden+from space的对象如果都存活,必须有足够的老生代空间存放这些对象。
  4. 这些都根据情况调整启动JVM的设置。
  5. 使用 Adaptive让JVM自动划分新生代和老生代。

Permanent Generation 永久代

  1. 该区域比较稳定,主要用于存放classloader信息,比如类信息和method信息。
  2. 缺省是 64M ,如果你的代码量很大,容易出现OutOfMemoryError: PermGen space 。
  3. 2G以上内存设置MaxPermSize为160M
  4. -XX:PermSize=128m -XX:MaxPermSize=160m


降低Full GC发生概率

  1. 为了降低Full GC发生概率,如果降低了老生代大小,那么 OutOfMemoryError 发生,Full GC概率反而会上升。
  2. 如果为了降低Full GC,增加老生代大小,执行时间可能会被延长。
  3. 必须寻找合适大小的老生代。
  4. 避免大的对象迁移到老生代。
  5. 减少迁移到老生代的对象数目


java.lang.OutOfMemoryError

  1. (1)在高负荷的情况下的却需要很大的内存,因此可以通过修改JVM参数来增加Java Heap Memory。
  2. (2)应用程序使用对象或者资源没有释放,导致内存消耗持续增加,关键采取OO封装边界方式,树立对象都有生命周期的基本习惯。
  3. (3)再一种也可能是对于第三方开源项目中资源释放了解不够导致使用以后资源没有释放(例如JDBC的ResultSet等)。


JVM参数

  1. -Xms, -Xmx—定义JVM的heap大小最小和最大值。
  1. -XX:NewSize— 定义年轻态的最小大小,Eden越大越好,但是越大响应有延迟。
  1. -Xmx2G -Xms1G -XX:NewSIze=512M (OldGen at least 1G)
  2. -Xmx3G -Xms1G -XX:NewSize=512M (OldGen at least 2G)
  3. Xmx4G -Xms2G -XX:NewSize=1G (OldGen at least 2.5G)
  4. -Xmx6G -Xms3G -XX:NewSize=2G (OldGen at least 3.5G)
  5. -Xmx8G -Xms4G -XX:NewSize=3G (OldGen at least 4.5G)

参数调整示意

  1. JAVA_OPTS="$JAVA_OPTS -server -Xss1280K -Xms1664m -Xmx1664m  -XX:MaxPermSize=128m -XX:SurvivorRatio=16  -XX:NewSize=1280m  -XX:MaxNewSize=1280m -XX:+DisableExplicitGC -XX:GCTimeRatio=2 -XX:ParallelGCThreads=4 -XX:+UseParNewGC -XX:MaxGCPauseMillis=2000 -XX:+UseConcMarkSweepGC  -XX:CMSInitiatingOccupancyFraction=80 -XX:+CMSClassUnloadingEnabled



Survivor大小

  1. NewSize / ( SurvivorRatio + 2)
  2. 如果SurvivorRatio =16, NewSize =1280m,那么S大小是70M。
  3. 太小,溢出的复制Collection进入老生代。
  4. 太大,闲置无用 浪费内存。
  5. 使用XX:+PrintTenuringDistribution  和-XX:+PrintGCDetails, -XX:+PrintHeapAtGC观察:
  6. 与 -XX:+UseAdaptiveSizePolicy 冲突


垃圾回收机制启动

  1. 垃圾回收机制不会频繁启动,因为机制一旦启动,造成应用程序停顿。
  2. 机制一般内存剩余5%左右启动,所以有现象:启动服务器,内存不断消耗,有多大内存消耗多大。
  3. 问题:如果服务器程序频繁触及5%底线,机制频繁启动,造成服务器慢..甚至死机。
  4. 根源:应用程序无限制频繁大量创建对象,消耗内存。


控制垃圾回收

  1. 带CMS参数的都是和并发回收相关的
  2. -XX:+UseParNewGC,对新生代采用多线程并行回收。
  3. CMSInitiatingOccupancyFraction=90说明年老代到90%满的时候开始执行对年老代的并发垃圾回收(CMS)
  4. 用jmap和jstack查看


串行 并行回收的区别

  1. 新生代 高吞吐量:
  2. -XX:+UseSerialGC
    -XX:+UseParallelGC
    -XX:+UseParNewGC
  3. 老生代 低暂停:
  4. -XX:+UseParallelOldGC
    -XX:+UseConcMarkSweepGC
  5. 相同点:GC时都暂停一切。
  6. 不同点:一个线程和多个线程同时GC

并行和CMS(Concurrent-Mark-Sweep)区别

  1. CMS步骤:
  2. - initial mark
    - concurrent marking
    - remark
    - concurrent sweeping
  3. 区别:CMS一个线程,并行多个线程
  4. CMS只是在1 3阶段暂停,而并行全部暂停。

Parallel GC 和 CMS GC

  1. 压实compaction是移除内存碎片,也就是移除已经分配的内存之间的空白空间。
  2. 在Parallel GC中,无论Full GC是否执行,压实总是被执行,会花费更多时间,不过在执行完Full GC后,内存也许再被使用时,会分配得快些,能够顺序分配了。
  3. CMS GC 并不执行压实,所以更快,碎片太多,没有空间放置大的需要连续空间的对象,“Concurrent mode failure”会发生。


并行和CMS配置

  1. -XX:UserParNewGC 适合于

新生代 (multiple GC threads)
-XX:+UseConcMarkSweepGC  适合于
  老生代 (one GC thread, freezes the JVM only during the initial mark and remark phases)
  -XX:InitiatingOccupancyFraction 80是表示CMS是在老生代接近满80%启动,如CPU空闲,可设定点一些。

-XX:+CMSIncrementalMode 用于CMS,不会让处理器Hold住整个并发phases  。

时间: 2024-11-05 12:08:20

JVM内存模型和性能优化的相关文章

Java内存模型及性能优化

最近在做一个项目的性能优化,遇到好多以前没有关注过的性能问题,一头雾水,今天做个笔记,简单记录下JVM相关的参数设置. 一.JVM内存模型 首先介绍下Java程序具体执行的过程: Java源代码文件(.java后缀)会被Java编译器编译为字节码文件(.class后缀): 由JVM中的类加载器加载各个类的字节码文件,加载完毕之后,交由JVM执行引擎执行: 在整个程序执行过程中,JVM会用一段空间来存储程序执行期间需要用到的数据和相关信息,这段空间一般被称作为Runtime Data Area(运

jvm的stack和heap,JVM内存模型,垃圾回收策略,分代收集,增量收集(转)

深入Java虚拟机:JVM中的Stack和Heap(转自:http://www.cnblogs.com/laoyangHJ/archive/2011/08/17/gc-Stack.html) 在JVM中,内存分为两个部分,Stack(栈)和Heap(堆),这里,我们从JVM的内存管理原理的角度来认识Stack和Heap,并通过这些原理认清Java中静态方法和静态属性的问题. 一般,JVM的内存分为两部分:Stack和Heap. Stack(栈)是JVM的内存指令区.Stack管理很简单,push

JVM内存模型及垃圾回收机制

JVM内存模型1.栈Java栈是与每一个线程关联的,JVM在创建每一个线程的时候,会分配一定的栈空间给线程.存储局部变量.引用.方法.返回值等.StackOverflowError:如果在线程执行的过程中,栈空间不够用,那么JVM就会抛出此异常,这种情况一般是死递归造成的.2.堆 Java中堆是由所有的线程共享的一块内存区域,堆用来保存各种JAVA对象,比如数组,线程对象等. 2.1堆的分代JVM堆一般分为三个部分:Young:年轻代Young区被划分为三部分,Eden区和两个大小严格相同的Su

JVM内存模型以及垃圾回收

原文:http://www.blogjava.net/freeman1984/archive/2011/03/08/345929.html JVM内存模型以及垃圾回收内存由 Perm 和 Heap 组成. 其中 Heap = {Old + NEW = { Eden , from, to } }具体可查看javavisualvm JVM内存模型中分两大块,一块是 NEW Generation, 另一块是Old Generation. 在New Generation中,有一个叫Eden的空间,主要是

Java学习之:JVM内存模型

一.文章来由 开始实习啦,实习转战Java开发工程师... 二.JVM内存模型总图 Java中通过多线程机制使得多个任务同时执行处理,所有的线程共享JVM内存区域main memory,而每个线程又单独的有自己的工作内存,当线程与内存区域进行交互时,数据从主存拷贝到工作内存,进而交由线程处理(操作码+操作数). 在之前,我们也已经提到,JVM的逻辑内存模型如下: 三.JVM内存模型详解 1.程序计数器 程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可

JVM内存模型 三

本文章节: 1.JMM简介 2.堆和栈 3.本机内存 4.防止内存泄漏 1.JMM简介 i.内存模型概述 Java平台自动集成了线程以及多处理器技术,这种集成程度比Java以前诞生的计算机语言要厉害很多,该语言针对多种异构平台的平台独立性而使用的多线程技术支持也是具有开拓性的一面,有时候在开发Java同步和线程安全要求很严格的程序时,往往容易混淆的一个概念就是内存模型.究竟什么是内存模型?内存模型描述了程序中各个变量(实例域.静态域和数组元素)之间的关系,以及在实际计算机系统中将变量存储到内存和

JVM理论:(一)JVM内存模型

一.JVM内存模型 1.程序计数器 线程私有,当前线程所执行的字节码的行号指示器,通过计数器来选取下条需要执行的字节码指令,分支.循环.跳转.异常处理.线程恢复等功能都依赖此功能,唯一没有规定OutOfMemoryError的区域,若执行的是Native方法,计数器值为空. 2.Java虚拟机栈 (1)线程私有,生命周期和线程相同. (2)栈与栈帧定义的区别: 每创建一个线程,虚拟机就会为这个线程创建一个虚拟机栈,每调用一个方法就会为每个方法生成一个栈帧,一个方法从调用到完成对应着一个栈帧在虚拟

深入学习JVM了解JVM内存模型

我们知道,计算机CPU和内存的交互是最频繁的,内存是我们的高速缓存区,用户磁盘和CPU的交互,而CPU运转速度越来越快,磁盘远远跟不上CPU的读写速度,才设计了内存,用户缓冲用户IO等待导致CPU的等待成本,但是随着CPU的发展,内存的读写速度也远远跟不上CPU的读写速度,因此,为了解决这一纠纷,CPU厂商在每颗CPU上加入了高速缓存,用来缓解这种症状. Jvm内存模型 虚拟机内存模型中定义的访问操作与物理计算机处理的基本一致! Java 中通过多线程机制使得多个任务同时执行处理,所有的线程共享

JVM内存模型与GC算法(简介)

JVM内存模型如上图,需要声明一点,这是<Java虚拟机规范(Java SE 7版)>规定的内容,实际区域由各JVM自己实现,所以可能略有不同.以下对各区域进行简短说明. 1.1程序计数器 程序计数器是众多编程语言都共有的一部分,作用是标示下一条需要执行的指令的位置,分支.循环.跳转.异常处理.线程恢复等基础功能都是依赖程序计数器完成的. 对于Java的多线程程序而言,不同的线程都是通过轮流获得cpu的时间片运行的,这符合计算机组成原理的基本概念,因此不同的线程之间需要不停的获得运行,挂起等待