Android OOM 排查与解决——图片加载优化

1、OOM 引起与表现

        在 Android 这种移动设备上,如果代码没有处理好,很容易引发内存持续占用与泄漏,导致 OOM(OutOfMemoryError) 异常,进而导致 App 程序 Crash 挂掉。

        在 Android 开发中,一个典型的 OOM 异常如下:

        
OOM 异常

        一旦碰上了这类错误,我们往往需要去排查内存了。导致 OOM 的一些情况比较常见,大多数情况下,大家可能遇到的都是同一种情况:

  • Activity 泄漏导致;
  • 层次庞大复杂的 View 视图导致;
  • 大量图片持续占用导致;
  • 其他资源持续未释放导致。

2、Android Studio 查看内存占用

        在 Android Studio 里面,我们可以在 Monitors 窗口中,实时对 App 内存进行监控,我们可以看出 App 的 HeapSize已经使用的内存大小剩余内存大小以及峰值变化。有了这些信息,我们可以在某个页面打开和关闭时进行监控,从而对比该页面占用内存变化,可以很方便的定位问题。

        
Monitors 查看内存占用

        在这张图中,如果已使用的内存大小(Allocated)接近到 HeapSize 的大小,App 将会处于非常危险的状态中,很有可能下一个操作就会直接导致 OOM,通过 Android Studio,我们可以防患于未然,在 Debug 阶段进行预防。

3、adb 查看内存占用

        adb 工具也是一个非常有用的工具,我们可以通过它来查看 App 内存占用。

3.1、查看 JVM 的 HeapSize 等参数

        通过命令 adb shell getprop dalvik.vm.heapsize 可以直接查看 Dalvik 虚拟机为 App 规定的最大 HeapSize

        
getprop dalvik.vm.heapsize

        一般来说,App 可达到的最大 HeapSize 为 dalvik.vm.heapgrowthlimit 所规定的大小。但是如果我们在 AndroidManifest.xml 中为 Application 添加 android:largeHeap="true" 属性,App 可达到的最大 HeapSize 则被调整为 dalvik.vm.heapsize 规定的值。

        虽然添加 android:largeHeap="true" 属性将大大降低 OOM 的概率,但除非万不得已的情况下,否则不要使用该属性。出现 OOM 后,我们首先应该排查整个 App,找出内存瓶颈予以解决。

3.2、查看 App 内存占用

        通过命令 adb shell dumpsys meminfo [package_name] 可以查看 App 所占用内存:

        
dumpsys meminfo 查看内存占用

        通过这个命令,App 所占资源情况一目了然,甚至我们可以看到整个 App 中 View 个数、Activity 个数——这对于排查 Activity 泄漏和优化 View 层级也是非常有帮助的。

4、图片加载导致 OOM

        而在一个 App 中,图片处理不恰当往往是 OOM 错误出现的元凶——因为 App 中所有图片动辄占用几十 M 的内存。如果我们能优先着手排查这一块,将会对 App 的内存优化带来 最直接最明显 的改观。而图片的不恰当处理操作一般有如下一些:

  • 直接加载 超大尺寸 图片;
  • 图片加载后 未及时释放
  • 在页面中,同时加载 非常多 的图片;

4.1、超大尺寸图片处理

        现在的手机摄像头像素比较高,摄制出来的照片尺寸非常大,比如在一款还算老旧的手机上面,拍摄的图片尺寸竟然达到了 2368 x 4224!因为采用 jpeg 格式的缘故,这张图片在磁盘上才1.9M,但如果我们不加任何处理,按原尺寸加载到内存中,占用的内存将会非常可观。

        所以,针对大图的加载,比较常用的方法是进行 DownSampling(向下采样),许多博客或技术站点对该方案有详细的描述,在此不再赘述,简单原理用代码表述如下:

public static int calcInSampleSize(
        int width, int height, int requestWidth, int requestHeight) {

    int inSampleSize = 1;
    if (requestWidth <= 0 || requestHeight <= 0) {
        return inSampleSize;
    }

    if (width > requestWidth || height > requestHeight) {
        int widthRatio = Math.round((float) width / (float) requestWidth);
        int heightRatio = Math.round((float) height / (float) requestHeight);

        inSampleSize = Math.min(widthRatio, heightRatio);
    }

    return inSampleSize;
}

public static Bitmap decodeBitmapFromUri(
    Context context, Uri uri, int requestWidth, int requestHeight) {

    BitmapFactory.Options options = getResourceOptions(context, uri);
    options.inSampleSize = calcInSampleSize(
            options.outWidth, options.outHeight, requestWidth, requestHeight);
    options.inJustDecodeBounds = false;

    // ...
    
    ContentResolver resolver = context.getContentResolver();
    Bitmap bitmap = BitmapFactory.decodeStream(
            resolver.openInputStream(uri), new Rect(), options);
    // ...

    return bitmap;
}

        这样,如果我们要加载一张图片到 View 上,我们可以通过 view.getMeasuredWidth() 和 view.getMeasuredHeight() 得到 View 的宽和高,然后按这个大小进行采样,得到的 Bitmap 将会是尺寸适合的图片,不会占用额外内存,图片在 View 上展示出来质量也比较高。

4.2、及时释放图片

        一般不要静态缓存图片,就算有缓存,也可以结合 LRU 机制来保证缓存图片的个数和占用内存。Android SDK 已经提供了 LruCache 类来实现 LRU 机制。

4.3、避免同时加载大量图片

        避免同一时间加载大量的图片,也可以为我们的内存优化提供不小的收益。比如,在一个 ScrollView 中有非常多的 ImageView,这时候,占用的内存往往非常客观,因为就算一些 View 我们在屏幕视野里面看不到,它还是持续占用内存。我们可以通过 RecyclerView 或者 ListView 来予以替换,从而达到内存优化的效果。

        在我的开发过程中,就遇到了这样一个例子。一个页面用 ScrollView 来布局,里面有 26 张左右的图片,这时候,整个 App 的内存占用长期达到了 90M 左右!一直徘徊在 OOM 边缘。在我把这个页面用 RecyclerView 替换掉 ScrollView 后,整个 App 内存竟然下降了 40M 之多!!!整个 App 变得非常顺滑。

5、采用开源库加载图片

        现在已经有非常多的图片加载库供我们使用了,比较流行的有:FrescoUniversal-Image-LoaderPicassoVolley 等等。这些开源库一般来说,对内存的优化已经比较全面了,比我们自己手工管理内存来的好。所以,可以根据项目的实际情况灵活选用。

        比如,我目前所使用的 Fresco 库,就可以灵活设定图片尺寸,避免加载大尺寸的图片(setResizeOptions):

public static void displayImage(DraweeView draweeView, Uri uri) {
    Size size = getAppropriateSize(draweeView);
    
    ImageRequest request = ImageRequestBuilder
            .newBuilderWithSource(uri)
            .setResizeOptions(new ResizeOptions(size.mWidth, size.mHeight))
            .setAutoRotateEnabled(true)
            .build();
    
    DraweeController controller = Fresco.newDraweeControllerBuilder()
            .setUri(uri)
            .setImageRequest(request)
            .setOldController(draweeView.getController())
            .build();
            
    draweeView.setController(controller);
}

private static Size getAppropriateSize(View view) {
    int width = view.getMeasuredWidth();
    int height = view.getMeasuredHeight();

    if (width <= 0 || height <= 0) {
        width = view.getWidth();
        height = view.getHeight();
    }

    Size size = MiscUtils.getScreenSize();
    if (width <= 0 || height <= 0 || width > size.mWidth || height > size.mHeight) {
        width = size.mWidth;
        height = size.mHeight;
    }

    return new Size(width, height);
}

        当然,我们还要在 ImagePipelineConfig 中开启 DownSamplingsetDownsampleEnabled(true)):


public static void initFresco(Context context) {
    ImagePipelineConfig config = ImagePipelineConfig
            .newBuilder(context)
            .setDownsampleEnabled(true)
            .build();
            
    Fresco.initialize(context, config);
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,900评论 25 707
  • 本文转载来源 http://www.csdn.net/article/2015-09-18/2825737/1 (...
    yoosir阅读 1,088评论 0 5
  • HereAndroid的内存优化是性能优化中很重要的一部分,而避免OOM又是内存优化中比较核心的一点。这是一篇关于...
    HarryXR阅读 3,807评论 1 24
  • 介绍自己负责的部分,如何实现的。 框架的搭建排查问题以及结解决方式兼容性保证性能优化上线之后模块导致crash的比...
    黄海佳阅读 13,147评论 6 350
  • 陀螺,陀螺,急速飞转的陀螺 皮鞭抽打着,火星四溅着,你是疯狂燃烧的自我 陀螺,陀螺,偏离的地点在你的左边 鞭子甩出...
    原朔阅读 131评论 0 1