GC那些事儿--Android内存优化第一弹

shengyu06 8年前
   <p style="text-align: center;"><img src="https://simg.open-open.com/show/d94d7a8639d6f1f610c5e3cfab55d333.png"></p>    <h2><strong>引言</strong></h2>    <p>接 App优化之内存优化(序) , 作为App优化系列中内存优化的一个小部分.</p>    <p>由于内存相关知识比较生涩, 内存优化中使用到的相关工具, 也有很多专有名词. 对Java内存管理, GC, Android内存管理, Dalvik/ART等知识有一个理论的认识, 可以让我们更好的使用这些工具, 分析内存问题.</p>    <p>据此, 我们就先从理论入手, 聊聊GC那些事儿.</p>    <h2><strong>1, 何为GC</strong></h2>    <p>GC 是 garbage collection 的缩写, 垃圾回收的意思. 也可以是 Garbage Collector, 也就是垃圾回收器.</p>    <h3><strong>1.1 垃圾回收器</strong></h3>    <p>我们先来解释下Garbage Collector(垃圾回收器).</p>    <p>内存管理, 一直是编程中的一个大的问题. 在较老的语言中, 例如C++语言中, 内存管理是显式的, 也就是说使用者自己申请内存使用, 自己释放内存. 这就是为什么C++语言中除了构造函数, 还有析构函数. 我们在创建对象的时候调用构造函数创建, 系统会在对象结束其作用域的时候调用析构函数, 我们需要做的就是在析构函数中释放掉我们申请的相关资源, 以便释放内存地址.</p>    <p>显然, 这种显式的由编程人员自己控制释放内存的方式很容易出问题, 忘了, 漏了, 都可能导致内存问题. 也不符合程序员要懒的特征.</p>    <p>故而, Java语言中引入了自动内存管理的机制, 也就是垃圾回收器. 大部分的现代面向对象语言, 也都是采用自动内存管理机制.</p>    <p>内存自动管理回收机制可以解决大部分, 但不是所有的内存问题, 这也是为什么我们要讨论内存泄露.</p>    <p>垃圾回收器的职责</p>    <p>垃圾回收器有三大职责:</p>    <ol>     <li> <p>分配内存;</p> </li>     <li> <p>确保任何被引用的对象保留在内存中;</p> </li>     <li> <p>回收 <strong>不能通过引用关系找到</strong> 的对象的内存.</p> </li>    </ol>    <p>垃圾回收的一般流程</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/81170253f14bb2ce7f9d1fa579def566.jpg"></p>    <p style="text-align: center;">gc process</p>    <h3><strong>1.2 相关概念</strong></h3>    <p><strong>垃圾回收(GC)</strong></p>    <p>垃圾回收器中有一个进程来做上面的这些事情, 这个进程查找我们的对象引用的关系并释放其内存, 这个进程就是garbage collection(垃圾回收), 也就是我们常说的GC.</p>    <p><strong>Heap和Stack</strong></p>    <p>简单说下:</p>    <ul>     <li>Heap内存是指java运行环境用来分配给对象和JRE类的内存. 是应用的内存空间.</li>     <li>Stack内存是相对于线程Thread而言的, 它保存线程中方法中短期存在的变量值和对Heap中对象的引用等.</li>     <li>Stack内存, 顾名思义, 是类Stack方式, 总是后进先出(LIFO)的.</li>     <li>我们通常说的GC的针对Heap内存的. 因为Stack内存相当于是随用随销的.</li>    </ul>    <p><img src="https://simg.open-open.com/show/6526f39c035d14acc7ddbbde50251c51.png"></p>    <p style="text-align: center;">heap&stack</p>    <p><strong>GC Root</strong></p>    <p>直译GC根, 我们姑且不译了吧.</p>    <p>所谓GC Root我们可以理解为是一个Heap内存之外的对象, 通常包括但不仅限于如下几种:</p>    <ul>     <li> <p>System Class <strong>系统</strong> Class Loader加载的类. 例如java运行环境中rt.jar中类, 比如java.util.* package中的类.</p> </li>     <li> <p>Thread 运行中的线程</p> </li>     <li> <p>JNI 中的本地/全局变量, 用户自定义的JNI代码或是JVM内部的.</p> </li>     <li> <p>Busy Monitor 任何调用了wait()或notify()方法, 或是同步化的(synchronized)的东西. 可以理解为同步监控器.</p> </li>     <li> <p>Java本地实例, 还在运行的Thread的stack中的方法创建的对象.</p> </li>    </ul>    <p><strong>活对象/垃圾</strong></p>    <p>如果这个对象是 <strong>引用可达</strong> 的, 则称之为 <strong>活的(live)</strong> , 反之, 如果这个对象 <strong>引用不可达</strong> , 则称之为 <strong>死的(dead)</strong> , 也可以称之为 <strong>垃圾(garbage)</strong> .</p>    <p>这个 <strong>引用可达与不可达</strong> 就是相对于GC Root来说的:</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/e8b71c081a23de23cf4a9882f1f408af.png"></p>    <p style="text-align: center;">gc-roots</p>    <h2><strong>2, Java的内存管理机制</strong></h2>    <h3><strong>2.1 关于JVM</strong></h3>    <p>我们平常在查看我们的java版本时, 你会发现:</p>    <pre>  $ java -version  java version "1.8.0_74"  Java(TM) SE Runtime Environment (build 1.8.0_74-b02)  Java HotSpot(TM) 64-Bit Server VM (build 25.74-b02, mixed mode)</pre>    <p>其中有个HotSpot VM的东西, 那么这个是什么呢? 和JVM有什么关系呢?</p>    <p>在此简单说下, 以便行文:</p>    <ul>     <li>JVM, Java虚拟机, 可以简单理解为一种技术思想, 虚拟技术理念.</li>     <li>HotSpot VM 是JVM的一种实现, 包含了服务器版和桌面应用程序版, 现时由Oracle维护并发布.</li>    </ul>    <p>我们当前使用的sun(oracle)的java版本(应该是1.3以上)都是内置的HotSpot VM实现. 所以接下来的分析也都是基于HotSpot VM的, 但是还是简称JVM.</p>    <h3><strong>2.2 JVM内存区域</strong></h3>    <p>JVM使用分代式的内存管理方式, 将Heap分成三代 --- 新生代, 老一代, 持久代.</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/fea48afff507b57bc696a430879440a8.png"></p>    <p style="text-align: center;">Hotspot heap structure</p>    <ul>     <li> <p>Young Generation</p>      <ul>       <li> <p>新生代.</p> </li>       <li> <p>所有new的对象.</p> </li>       <li> <p>该区域的内存管理使用minor garbage collection(小GC).</p> </li>       <li> <p>更进一步分成Eden space, Survivor 0 和 Survivor 1 三个部分.</p> </li>      </ul> </li>     <li> <p>Old Generation</p>      <ul>       <li> <p>老年区.</p> </li>       <li> <p>新生代中执行小粒度的GC幸存下来的"老"对象.</p> </li>       <li> <p>该区域的内存管理使用major garbage collection(大GC).</p> </li>      </ul> </li>     <li> <p>Permanent Generation</p>      <ul>       <li> <p>持久代.</p> </li>       <li> <p>包含应用的类/方法信息, 以及JRE库的类和方法信息.</p> </li>      </ul> </li>    </ul>    <p>小GC执行非常频繁, 而且速度特别快.</p>    <p>大GC一般会比小GC慢十倍以上.</p>    <p>大小GC都会发出"Stop the World"事件, 也就是说中断程序运行, 直至GC完成. 这也是我们在App优化之消除卡顿中为什么说频繁GC会造成用户感知卡顿.</p>    <h2><strong>3, GC的流程</strong></h2>    <p>了解了内存Heap的几个区域, 我们再来看下垃圾收集器是怎么利用这几个区域来管理内存和回收垃圾的.</p>    <p>1. 创建新的对象</p>    <p>每当我们使用new创建一个对象时, 这个对象会被分配到 <strong>新生代</strong> 的 <strong>Eden</strong> 区域:</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/87b70096f2e98aa3d3b6c7232fb7341c.jpg"></p>    <p style="text-align: center;">object allocation</p>    <p><strong>2. 当Eden区域满时</strong></p>    <p>当Eden区域内存被分配完时, <strong>小GC</strong> 程序被触发:</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/837ea2f1cdd5070f928a0c7e0f223283.png"></p>    <p style="text-align: center;">Eden filling</p>    <p>引用可达的对象会移到Survivor(幸存者)区域-- <strong>S0</strong> , 然后清空Eden区域, 此时 <strong>引用不可达的对象</strong> 会直接删除, 内存回收, 如下:</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/8810bb9c98e6c458308e5c630f0a1d21.png"></p>    <p style="text-align: center;">aged</p>    <p><strong>3. Eden再次满时</strong></p>    <p>当Eden区域再次分配完后, <strong>小GC</strong> 执行, <strong>引用可达的对象</strong> 会移到Survivor(幸存者)区域, 而 <strong>引用不可达的对象</strong> 会跟随Eden的清空而删除回收.</p>    <p>需要注意的是, 这次 <strong>引用可达的对象</strong> 移动到的是 <strong>S1</strong> 的幸存者区.</p>    <p>而且, S0区域也会执行 <strong>小GC</strong> , 将其中还 <strong>引用可达的对象</strong> 移动到S1区, 且年龄+1. 然后清空S0, 回收其中 <strong>引用不可达的对象</strong> .</p>    <p>此时, 所有 <strong>引用可达的对象</strong> 都在S1区, 且S1区的对象存在不同的年龄. 如下:</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/5e0a292ac2594fea2dc2c0d564cc9bc1.png"></p>    <p style="text-align: center;">next filling</p>    <p>当Eden第三次满时, S0和S1的角色互换了:</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/fb6a7c1ce0df3c22b2e7f32b4389a42a.png"></p>    <p style="text-align: center;">s0s1</p>    <p>依此循环.</p>    <p><strong>4. 当Survivor区的对象年龄达到"老年线"时</strong></p>    <p>上面1~3循环, Survivor区的对象年龄也会持续增长, 当其中某些对象年龄达到"老年线", 例如8岁时, 它们会"晋升"到 <strong>老年区</strong> .</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/bc80ed810528b9f27e7146629d4dd675.png"></p>    <p style="text-align: center;">old aged</p>    <p>如此1~4步重复, 大体流程是这样的</p>    <p><img src="https://simg.open-open.com/show/2ab9e9b2ef67d59041f6bc571f1241e7.png"></p>    <p style="text-align: center;">gc flow</p>    <h2>参考</h2>    <ul>     <li><a href="/misc/goto?guid=4959720736467568901" rel="nofollow,noindex">http://www.oracle.com/webfolder/technetwork/tutorials/obe/java/gc01/index.html</a></li>     <li><a href="/misc/goto?guid=4959720736561204176" rel="nofollow,noindex">http://www.ibm.com/support/knowledgecenter/SS3KLZ/com.ibm.java.diagnostics.memory.analyzer.doc/gcroots.html</a></li>     <li><a href="/misc/goto?guid=4959720736644508882" rel="nofollow,noindex">http://www.infoq.com/cn/articles/jvm-memory-collection</a></li>     <li><a href="/misc/goto?guid=4959720736723596892" rel="nofollow,noindex">http://www.journaldev.com/4098/java-heap-space-vs-stack-memory</a></li>    </ul>    <p> </p>    <p> </p>    <p>来自:http://www.jianshu.com/p/5db05db4f5ab</p>    <p> </p>