Android 开发——JVM复习小结

方法区域在虚拟机启动时创建.

如果方法区中的内存无法满足分配请求,Java 虚拟机将引发 OutOfMemoryError.

运行时常量池

属于方法区的一部分,用于存放编译期生成的各种字面量和符号引用.

编译期和运行期(String 的 intern() )都可以将常量放入池中.

内存有限,无法申请时排除OutOfMemoryError

创建类或接口时,如果运行时常量池的构造需要的内存多于 Java 虚拟机的方法区中可用的内存,则 Java 虚拟机将引发一个 OutOfMemoryError.

直接内存

非虚拟机运行时数据区的部分

在 JDK 1.4 中新加入 NIO (New Input/Output) 类,引入了一种基于通道(Channel)和缓存(Buffer)的 I/O 方式,它可以使用 Native 函数库直接分配堆外内存,然后通过一个存储在 Java 堆中的 DirectByteBuffer 对象作为这块内存的引用进行操作。可以避免在 Java 堆和 Native 堆中来回的数据耗时操作。 OutOfMemoryError:会受到本机内存限制,如果内存区域总和大于物理内存限制从而导致动态扩展时出现该异常。

垃圾回收器与内存分配策略


程序计数器,虚拟机栈和本地方法栈3个区域随线程生灭,栈中的栈针随着方法的进入和退出执行出栈和入栈的操作.

Java堆和方法区不一样,一个接口中的多个实现类需要的内存可能不一样,一个方法中的多个分支需要的内存可能也不一样,我们只有在程序处于运行期才知道哪些对象会创建,这部分内存的分配和回收都是动态的,垃圾回收所关注的就是这部分内存.

判断对象是否需要回收

引用计数法

每个对象有一个引用计数器,当对象被引用一次则计数器加1,当对应引用失效1次则计数器减1,对于计数器为0的对象意味着是垃圾对象,可以被GC回收.

缺点: Java堆中保持相互引用的对象无法回收,难以解决循环引用问题

可达性算法

从GC Roots作为起点开始搜索,从这些节点出发所走过的路径称为引用链. 那么整个链中的对象就是活对象.

对于GC Roots无法到达的对象随时可能被GC回收.

可作为 GC Roots 的对象:

  • 虚拟机栈(栈帧中的本地变量表)中引用的对象

  • 方法区中静态变量和常量引用的对象

  • 本地方法栈中 JNI(即一般说的 Native 方法) 引用的对象

引用

前面的算法判断存活都与’引用’有关.

下面4种引用,强度依次递减.

强引用

类似于Object obj = new Object();创建的 obj指向Object实例所在的堆空间.

强引用是使用最普遍的引用。如果一个对象具有强引用,那么垃圾回收器绝不会回收它

强引用特点:

  • 强引用可以直接访问目标对象

  • 只要有引用变量存在,垃圾回收器永远不会回收。JVM即使抛出OOM异常,也不会回收强引用所指向的对象。

  • 强引用可能导致内存泄漏问题

软引用

可以通过java.lang.ref.SoftReference类实现软引用.在系统要发生内存溢出之前,将会把这些对象列进回收范围中进行二次回收.

一个持有软引用的对象,不会被JVM很快回收,JVM会判断当前堆的使用情况来判断何时回收.

当堆的使用率临近阈值时,才回去回收软引用的对象.

软引用主要用来实现类似缓存的功能,在内存足够的情况下直接通过软引用取值,无需从繁忙的真实来源查询数据,提升速度;

当内存不足时,自动删除这部分缓存数据,从真正的来源查询这些数据。使用软引用能防止内存泄露,增强程序的健壮性。

弱引用

在java中,可以用java.lang.ref.WeakReference实例来保存对一个Java对象的弱引用。

在系统GC时,不管系统堆空间是否足够,都会将对象进行回收.

虚引用

可以通过PhantomReference 类实现虚引用. 无法通过虚引用获取一个对象的实例,为一个对象设置虚引用关联的唯一目的就是能在这个对象被收集器回收时收到一个系统通知。

一个持有虚引用的对象和没有引用几乎是一样的,随时可能被垃圾回收器回收.

它的作用在于检测对象是否已经从内存中删除,跟踪垃圾回收过程.

垃圾回收算法

标记清除法

标记清除法是最基础的收集算法 ,它分为"标记"和"清除"两个阶段:

首先标记出需要回收的对象,在标记完成后统一回收掉被标记的对象,它的标记过程其实就是前面的可达性分析算法中判定垃圾对象的标记过程.

  • 回收前:

  • 回收后:

缺点:

  • 标记和清除的效率都不高

  • 标记清除后会产生大量不连续的内存碎片,空间碎片太多可能会导致需要分配较大对象时无法找到足够的连续内存而不得不触发另一次垃圾回收操作.

复制算法

它将内存按容量分为大小相等的两块,每次只使用其中的一块,当这一块的内存用完了,就将还存货的对象复制到另一块内存上面 , 然后再把已使用的内存空间一次清理掉.

优点:

  • 每次只对一块内存进行回收,运行高效

  • 只需移动栈顶的指针,按顺序分配内存即可,实现简单

  • 内存回收时不用考虑内存碎片的出现

缺点:

  • 可一次分配的内存缩小了一半

  • 空间利用率下降

因为大多数新生代对象都不会熬过第一次 GC。所以没必要 1 : 1 划分空间。可以分一块较大的 Eden 空间和两块较小的 Survivor 空间,每次使用 Eden 空间和其中一块 Survivor。当回收时,将 Eden 和 Survivor 中还存活的对象一次性复制到另一块 Survivor 上,最后清理 Eden 和 Survivor 空间。大小比例一般是 8 : 1 : 1,每次浪费 10% 的 Survivor 空间。但是这里有一个问题就是如果存活的大于 10% 怎么办?这里采用一种分配担保策略:多出来的对象直接进入老年代。

  • 回收前:

  • 回收后:

标记整理法

复制算法比较适合于新生代,在老年代中,对象存活率比较高,如果执行较多的复制操作,效率将会变低,所以老年代一般会选用其他算法,如标记—整理算法。

该算法标记过程和标记清除法中的标记过程一样,然后把存活对象移到内存的一端,然后直接清理掉端边界以外的内存。

  • 回收前:

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

下面是辛苦给大家整理的学习路线

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!**

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

[外链图片转存中…(img-LF1RHggp-1713756859785)]

最后

下面是辛苦给大家整理的学习路线

[外链图片转存中…(img-Qo1f5FbT-1713756859787)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 21
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值