对象的内存分配,主要是在java堆上分配(有可能经过JIT编译后被拆为标量类型并间接地在栈上分配),如果启动了本地线程分配缓冲,将按线程优先在TLAB上分配。少数情况下也是直接分配到老年代,分配规则不是固定的,细节还是取决于垃圾收集器的组合,以及虚拟机和内存相关参数的配置。

  JVM 堆中分为 新生代 老年代 永久代,新生代分为 Eden区,SurvivorFrom区域和SurvivorTo区域

         

关于这几个概念的说明,可以参考 :http://ju.outofmemory.cn/entry/346964

普遍的分配规则
  • 对象优先在Eden分配
  • 大对象直接进入老年代
  • 长期存活对象进入老年代
  • 动态对象年龄判定
  • 空间分配担保

优先分配在Eden区

  大多数情况下,对象在新生代Eden区中分配。当Eden区中内存空间不足的时候,会进行一次Minor GC。

  Minor GC 与 Full GC (Major GC)
  Minor GC:指 发生在新生代的垃圾收集动作,因为新生代对象 "朝生夕死" 的特性,所以Minor GC非常频繁,一般回收的速度也快。
  Full GC:指发生在老年代的垃圾收集动作,Full GC的动作速度比较慢

大对象直接进入老年代

  大对象指的是,需要大量连续空间的java对象(例如数组和长字符串)。可以设置虚拟机参数:-XX:PretenureSizeThreshold  ,令大于这个参数的设置值的对象直接进入 老年代(-XX:PretenureSizeThreshold 参数只对 Serial 和 ParNew收集器有效,Parallel Scavenge收集器不识别这个参数)。

 
长期存活的对象进入老年代
  虚拟机给每个对象定义了一个年龄计数器(Age),如果对象在Eden分配了空间,当Minor GC发生后仍然存活,而且能够被Survivor容纳,则会移动到Survivor,并且Age +1,对象在Survivor区中每经历一次Minor GC,n年龄计数器则会+1,当达到一定程度(默认是15,这个阀值由参数 -XX:MaxTenuringThreshold 来设置),这个对象则会晋升到老年代中。
 
动态对象年龄判定
    为了适应各种应用程序的内存情况,虚拟机并不要求java对象的年龄达到MaxTenuringThreshold才会将其移入到老年代中。当在Survivor区内相同年龄的所有对象大小总和大于Survivor空间的一半时,这些对象将会移入到老年代中,而无需等待年龄达到 MaxTenuringThreshold 设置的值。
 
空间分配担保
    当发生Minor GC的时候,虚拟机将会检测之前每次晋升到老年代对象的平均大小是否大于老年代的剩余空间大小。(理解为根据之前晋升到老年代对象的平均大小的 "经验" 来判断老年代的的剩余空间是否足够存储那些即将晋升到老年代的对象)
  • 如果大于则进行一次Full GC
  • 如果小于则看 HandlePromotionFailure 设置是否允许担保失败,如果允许担保失败,则进行Minor GC,如果不允许,则再进行一次Full GC (大部分是打开允许的,为了避免频繁的 Full GC)。

JVM 内存分配和回收策略的更多相关文章

  1. jvm内存分配和回收策略

    在上一篇中,已经介绍了内存结构是什么样的. 这篇来介绍一下 内存是怎么分配的,和怎么回收的.(基本取自<深入理解Java虚拟机>一书) java技术体系中所提倡的自动内存管理最终可以归结为 ...

  2. A4. JVM 内存分配及回收策略

    [概述] Java 技术体系中所提倡的自动内存管理最终可以归结为自动化地解决两个问题:给对象分配内存以及回收分配给对象的内存. 对象的内存分配,往大方向讲,就是在堆上分配,对象主要分配在新生代的 Ed ...

  3. JVM——内存分配与回收策略

    1.对象优先在Eden区分配 大多数情况下,对象在新生代Eden区分配.当Eden区没有足够的空间进行分配时,虚拟机将发起一次Minor GC. 虚拟机提供了 -XX:+PrintGCDetails这 ...

  4. JVM内存分配与回收策略

    对象优先在Eden分配 大多数情况下,对象在新生代Eden区中分配. 当Eden区没有足够空间进行分配时,虚拟机将发起一次Minor GC. Minor GC:新生代GC,指发生在新生代的垃圾收集动作 ...

  5. Java虚拟机垃圾回收:内存分配与回收策略 方法区垃圾回收 以及 JVM垃圾回收的调优方法

    在<Java对象在Java虚拟机中的创建过程>了解到对象创建的内存分配,在<Java内存区域 JVM运行时数据区>中了解到各数据区有些什么特点.以及相关参数的调整,在<J ...

  6. JVM垃圾回收器、内存分配与回收策略

    新生代垃圾收集器 1. Serial收集器 serial收集器即串行收集器,是一个单线程收集器. 串行收集器在进行垃圾回收时只使用一个CPU或一条收集线程去完成垃圾回收工作,并且会暂停其他的工作线程( ...

  7. JVM学习十 -(复习)内存分配与回收策略

    内存分配与回收策略 对象的内存分配,就是在堆上分配(也可能经过 JIT 编译后被拆散为标量类型并间接在栈上分配),对象主要分配在新生代的 Eden 区上,少数情况下可能直接分配在老年代,分配规则不固定 ...

  8. 最简单例子图解JVM内存分配和回收

    一.简介 JVM采用分代垃圾回收.在JVM的内存空间中把堆空间分为年老代和年轻代.将大量(据说是90%以上)创建了没多久就会消亡的对象存储在年轻代,而年老代中存放生命周期长久的实例对象.年轻代中又被分 ...

  9. 最简单例子图解JVM内存分配和回收(转)

    本文转自http://ifeve.com/a-simple-example-demo-jvm-allocation-and-gc/ http://www.idouba.net/a-simple-exa ...

随机推荐

  1. SQL Server 脚本跟踪

    1.查询 DataBasesID select db_id('regdatas') 2.获取进程ID 3.过滤定位

  2. Spark Mllib里的分层抽样(使用map作为分层抽样的数据标记)

    不多说,直接上干货! 具体,见 Spark Mllib机器学习实战的第4章 Mllib基本数据类型和Mllib数理统计

  3. NeatUpload 的使用

    1 <httpModules> 2 <add name="UploadHttpModule" type="Brettle.Web.NeatUpload. ...

  4. SpringMVC项目的快速搭建

    Spring MVC提供了一个DispatcherServlet来开发Web应用.在Servlet2.5及2以下的时候只要在web.xml下配置<servlet>元素即可. 在Servle ...

  5. Visual Studio 更改护眼颜色

  6. docker制作共享jdk的tomcat镜像

    FROM centos:7.4.1708 #挂载宿主机jdk到容器,节省空间 MAINTAINER huqiang:2018/10/12 ENV VERSION=8.5.34 ENV CATALINA ...

  7. 使用CreateProcess函数运行其他程序

    为了便于控制通过脚本运行的程序,可以使用win32process模块中的CreateProcess()函数创建一个运行相应程序的进程.其函数原型如下.CreateProcess(appName, co ...

  8. linux 命令——58 ss(转)

    telnet 命令通常用来远程登录.telnet程序是基于TELNET协议的远程登录客户端程序.Telnet协议是TCP/IP协议族中的一员,是 Internet远程登陆服务的标准协议和主要方式.它为 ...

  9. JSON.parse()与JSON.stringify()

    JSON.parse() 将字符串转成JSON 举个例子 var str = '{"name":"cn","age":"2&quo ...

  10. Codeforces Round #324 (Div. 2) A B C D E

    A,水题不多说. #include<bits/stdc++.h> using namespace std; //#define LOCAL int main() { #ifdef LOCA ...