当前位置: 首页 > article >正文

Android的虚拟机垃圾回收解析,简单明了看完通透

Android虚拟机垃圾回收原理

Android虚拟机使用的是基于Java的Dalvik虚拟机,它的垃圾回收机制和Java虚拟机的垃圾回收机制类似。Dalvik虚拟机中的垃圾回收分为分代垃圾回收和并发垃圾回收两种方式。

首先,Dalvik虚拟机使用分代垃圾回收机制,将所有Java对象分为Young Generation、Old Generation和Permanent Generations三代。当一个对象被创建时,它被分配到Young Generation。在Young Generation中的对象,当内存不足时,就会触发一次Minor GC(年轻代垃圾回收)。Minor GC只扫描年轻代中的对象,清理出所有无法到达的对象,并将剩余的对象移动到Old Generation。

在Old Generation中的对象,当内存不足时,就会触发一次Major GC(老年代垃圾回收)。Major GC会扫描整个堆中的所有对象,清理出所有无法到达的对象,并将剩余的对象保留在Old Generation中。Permanent Generation是用于存放静态对象和常量的一块区域,不参与垃圾回收。

除了分代垃圾回收之外,Dalvik虚拟机还使用了基于标记-清除算法的并发垃圾回收机制。这种回收机制可以在系统正常运行时,进行垃圾回收操作。首先是标记阶段,通过标记所有无法到达的对象。然后在清除阶段回收无法到达的对象。

Java内存和非Java内存的垃圾回收

Android虚拟机垃圾回收机制由两个部分组成

Java内存的垃圾回收

Java内存的垃圾回收主要涉及分代垃圾回收和并发垃圾回收。

分代垃圾回收:Java的分代垃圾回收机制将内存分成了不同的区域,每个区域负责管理不同年龄的对象。在年轻代区域,垃圾回收不会引起程序停顿;在老年代区域,垃圾回收可能会导致程序停顿。当某个区域的内存使用超过了其限制时,就会触发垃圾回收。

并发垃圾回收:并发垃圾回收是在应用程序正常运行时,与操作系统同时运行的垃圾回收机制。在并发垃圾回收期间,虚拟机会启动一个线程来标记和清理无用对象,以释放内存空间。

非Java内存的垃圾回收

除了Java对象,Android应用程序还使用大量的非Java内存,如Bitmap、JNI对象和资源等。由于这些对象不受Java垃圾回收机制的管理,因此需要专门的垃圾回收机制来进行处理。

在非Java内存的垃圾回收中,主要涉及Bitmap的回收。Bitmap是一种占用大量内存的对象,使用不当容易引起内存问题。Android提供了一种bitmap.recycle()方法来释放Bitmap对象所占用的内存空间。如果在应用退出前手动将所有Bitmap回收,则可以避免内存泄漏和应用崩溃等问题。

总的来说,Android虚拟机垃圾回收机制是通过Java内存和非Java内存的垃圾回收机制来有效地管理内存,提高系统性能和稳定性。在实际应用中,程序员需要根据实际需求来合理地使用内存和垃圾回收机制,以避免出现内存泄漏和应用崩溃等问题。

基于Android虚拟机的垃圾回收实战代码

java
public class MainActivity extends AppCompatActivity {
​
    private String[] arr = new String[10000];
    private TextView textView;
​
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
​
        textView = findViewById(R.id.textView);
​
        for (int i=0; i<10000; i++) {
            arr[i] = Integer.toString(i);
        }
    }
​
    @Override
    protected void onResume() {
        super.onResume();
        updateUI();
    }
​
    private void updateUI() {
        StringBuilder stringBuilder = new StringBuilder();
​
        for (String s : arr) {
            stringBuilder.append(s).append("\n");
        }
​
        textView.setText(stringBuilder.toString());
    }
​
    @Override
    protected void onDestroy() {
        super.onDestroy();
​
        //手动调用垃圾回收机制
        System.gc();
    }
}

这个代码可以创建一个大小为10000的字符串数组,然后在onResume方法中对UI进行更新,最后在onDestroy方法中手动调用垃圾回收机制。在这个例子中,每次更新UI时,都会创建一个新的StringBuilder对象并通过循环将数组中的字符串连接起来。如果没有手动调用垃圾回收机制,这些未使用的StringBuilder对象和字符串对象将会一直占用内存,可能会导致内存泄漏和应用崩溃。通过手动调用垃圾回收机制,可以清除这些无法到达的对象,释放内存空间,保证应用的稳定性和性能。

总结

Android虚拟机垃圾回收机制的运作过程主要分为以下几个阶段

  • 内存分配阶段:当应用需要加载新的类或创建新的对象时,Android虚拟机会根据需要动态分配内存空间。
  • 分代回收阶段:Android虚拟机采用分代垃圾回收机制,将内存分为不同的代,根据不同代的对象生命周期,选择不同的垃圾回收策略进行回收,以优化内存的使用。
  • 并发标记阶段:应用程序正常运行时,Android虚拟机会启动一个线程用于对堆内存进行并发标记,标记存活对象,以便后续的清理操作。
  • 并发清理阶段:在并发标记阶段结束之后,虚拟机会启动一个线程来清理不可达对象,以回收内存空间。
  • 物理内存压缩阶段:当内存占用过高时,Android虚拟机会启动一个物理内存压缩阶段来释放内存空间,减少内存碎片。

http://www.kler.cn/a/4448.html

相关文章:

  • WEB 攻防-通用漏-XSS 跨站脚本攻击-反射型/存储型/DOMBEEF-XSS
  • 【算法】图解两个链表相交的一系列问题
  • Laravel 中 Cache::remember 的基本用途
  • MySQL主从:如何处理“Got Fatal Error 1236”或 MY-013114 错误(percona译文)
  • 网络层协议-----IP协议
  • 如何在Jupyter中快速切换Anaconda里不同的虚拟环境
  • Jython
  • 1652_MIT 6.828 shell例程重定向的实现分析
  • mysql隐式转换 “undefined“字符串匹配到mysql int类型0值字段
  • MySQL——锁
  • K8s CrashLoopBackOff 如何排障?
  • 如何注册Shopify商店
  • 汇编语言(1)——寄存器
  • 编写python的smtplib库发送邮件代码(简洁-原创)
  • js中generator详解
  • C++之引用
  • 类ChatGPT平台推荐【国内访问ChatGPT4】
  • eNSP 本地AAA配置实验
  • <class ‘sensor_msgs.msg._Image.Image‘>原理
  • Apache iotdb-web-workbench 认证绕过漏洞 CVE-2023-24829
  • Exception 和 Error
  • Pytorch梯度下降——up主:刘二大人《PyTorch深度学习实践》
  • 【HTB】Responder思路——Responder抓取ntlmhash、远程文件包含、远程代码执行、evil-winrm连接
  • 文件操作练习
  • Python中关于字典和Counter()的两点区别
  • PCB模块化设计06——HDMI接口PCB布局布线设计规范