Android性能优化案例(3)内存抖动和泄漏的优化

2、详细书写源码的追踪,源码截图,绘制类的结构图,尽量详细地解释原理的探索过程
3、提供Github 的 可运行的Demo工程,但是我所提供代码,更多是提供思路,抛砖引玉,请酌情cv
4、集合整理原理探索过程中的一些坑,或者demo的运行过程中的注意事项
5、用gif图,最直观地展示demo运行效果**

如果觉得细节太细,直接跳过看结论即可。
本人能力有限,如若发现描述不当之处,欢迎留言批评指正。

作者:波澜步惊
链接:https://www.jianshu.com/p/535ea0d06e2b
学到老活到老,路漫漫其修远兮。与众君共勉 !

本文接上一篇;APP卡顿优化


正文大纲

  • jvm内存管理常识
  • 检测以及处理内存抖动
  • 检测以及处理内存泄漏

正文

jvm内存管理常识

  • LMK (LowMemoryKill)机制
    android底层会在系统内存告急的时候,按照一定规则杀死一些进程来满足其他进程的内存需要。其中 消耗内存的高低就是其中一项指标,所以,优化app的内存占用,能够有效降低app被系统杀死的概率。
  • GC STW机制
    GC,垃圾回收进程,在GC线程执行任务的时候,会存在一个 STW (stop the world) 机制,他就会把其他所有线程都挂起。如果GC非常频繁地调用,那就会导致主线程不流畅,给用户的感觉就是卡顿
  • 内存抖动频繁引起OOM
    内存抖动太频繁,导致大量对象频繁创建和销毁,会产生大量不连续的内存空间,如果此时有一个大对象需要申请内存,就有可能申请失败,导致OOM内存溢出
  • 一句话解释 内存泄漏
    生命周期的对象持有生命周期对象的强引用,在生命周期对象需要回收的时候发现不能被回收,视为泄漏
  • GC回收 可达性分析
  • GC线程判定 一个对象是不是可以回收,是根据可达性分析算法,计算GcRoot,从GcRoot向下搜索,把GcRoot没有直接关联的对象全部作为垃圾来回收。
  • 强软弱虚四大引用
    强和虚自不必说。强 最常见,没有特殊处理的都是强引用(包括,匿名内部类会持有外部类的强引用)。虚引用没什么用,不予讨论。
    软引用,用来定义一些还有用,但是不是必须的对象,使用SoftRefrence<T>修饰,在内存紧张的时候,GC回收之后,使用SoftRefrence<T>修饰,如果系统还有足够的内存可用,那么软引用关联的对象就不会被回收。如果不足,则回收软引用关联的对象。
    弱引用(WeakRefrence<T>),比软引用更弱一些,只要GC触发,弱引用关联的对象就会被回收。
    注意,使用软和弱引用,要判定关联对象是否为空。

检测以及处理内存抖动

我们使用s开发,平时我们运行app,一般会点 RunApp,但是还有另一个选择, 那就是 profileApp, 运行app起来之后,会在as下方看到profile 窗口

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

点击之后,as下方会出现profile,图中会显示网络,内存和cpu使用情况

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

如果内存的图中抖动得非常明显,比如像这样的心电图一样:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

那就说明非常明显存在内存抖动,急需处理:

点击内存图形区域之后,就能看到详细的内存变化情况,以及内存分配情况:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

这里有个坑

如果你从图形中观察到,内存走势平稳,并没有出现上满模拟抖动的图中那么夸张,是不是就不存在内存抖动呢?并不是。因为我们的gc,是在内存不可用的情况下才会去回收内存,如果app占用内存一直比较少,没有触及gc的临界值,那么就不会出现断崖式下跌. 那么这样就观察不出内存抖动了,怎么办呢?

解决方法
在8.0以下的安卓手机上,在下方的位置上会出现一个Record按钮(如果是8.0以上,你可以直接用拖拽的方式来截取一段内存record):

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

点击它,一段时间之后,再点一下:你就能在下方发现一张表格:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

这张表格代表的是,你Record这段时间之内创建的对象,点击一下第二列Allocations,对创建的数量进行排序,找出创建次数最多的对象:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

然后,点击排行第一的String之后,会在右方看到:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

然后点击其中的一个,又会看到一个新的窗口:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

到此为止,就找到了创建对象元凶,以这个为线索,找到你们自己包名下的类和方法,确定是我们自己的代码在不合理地创建对象.

再往后,就是根据各自的业务代码去做优化了,记住一个宗旨:不要让代码干多余的事。如果是我们调用了系统的api导致了不合理地大量对象的创建,那么就要考虑这个系统API为什么会这样创建对象,有没有其他方法避免吗,从业务代码层来合理使用这个api,实在不行再考虑自定义api或者换个系统api。

在我们做了一次优化之后,再profile运行一次app,再重复上面的过程。以此类推,直到内存抖动达到理想状态。

总结

优化内存抖动,核心就是防止频繁创建对象。常见的反面教材就是:循环中创建对象,大量调用的api中创建对象。而优化的主要手段,就是对象复用,常见的手段是:对象池,像是 Handler的Message 单链表池,Glide的bitmap池等。


检测以及处理内存泄漏

经典案例:处理handler异步任务导致的内存泄漏方法

  • 在Activity的onDestroy中移除所有的任务

@Override
protected void onDestroy() {
super.onDestroy();
handler.removeCallbacksAndMessages(null);//移除所有任务
}

  • 使用静态内部类 + Activity弱引用的方式

MyHandler handler = new MyHandler(this);
private static class MyHandler extends Handler {
WeakReference activityWeakReference;

MyHandler(Activity activity) {
activityWeakReference = new WeakReference<>(activity);
}

@Override
public void handleMessage(Message msg) {
//在执行任务的时候,判断弱引用所关联的对象是否为空,能在对象已经被回收的情况下,不去执行不必要的任务
switch (msg.what) {
case 1:
if (activityWeakReference.get() != null) {
//TODO
}
}
}
}

工具的使用

依然是profileApp,先用profile看出内存的变化情况。

  • 问:如何判断内存泄漏?
    答:内存泄漏是精细功夫,不能全盘观察,只能凭借profile的内存变化来推测
    比如,打开app之后内存一路飙升,直到超出app能够使用的最大内存,app崩溃,,这是最明显的。
    又比如,你反复打开关闭某一个界面,发现内存的稳定线(内存稳定之后,内存占用值)随着每一次的打开关闭,都在提高,这说明,这一个界面上存在泄漏,有对象无法被回收。

上一章节使用profile 最多是了解到 哪些对象的创建和回收引起了内存抖动,但是,涉及到泄漏,只通过profile尚且不能知道是哪个类持有了希望被回收的对象的强引用.

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

深知大多数初中级安卓工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

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

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Android)
img

最后

下面是辛苦给大家整理的学习路线,有需要的可以点击这里免费获取

oid)**
[外链图片转存中…(img-Eg4M3hiq-1710568021528)]

最后

下面是辛苦给大家整理的学习路线,有需要的可以点击这里免费获取
[外链图片转存中…(img-8JgVgpDs-1710568021528)]

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值