基础Android技术知识Android开发

Android bitmap config你理解对了吗?

2017-06-14  本文已影响135人  Jamling

在写本文之前,我还以为自己对android的bitmap很熟悉,直到自己亲手写代码实践,才发现自己错了很多年。真是汗颜啊!

Bitmap Config

首先,根据Android API 25的文档简要说明一下Android的Bitmap.Config以下4个选项

bitmap.config ALPHA_8 ARGB_4444 ARGB_8888 RGB_565
bytes/pixel 1 byte 2 byte 4 byte 2 byte
alpha channel 8 bit 4 bit 8 bit not support

PNG 格式

其次,简要说一下png格式

bitmap内存占用计算

第三,简要说一下bitmap占用的内存

Android中bitmap的内存占用是跟图片的尺寸(高和宽)相关。一张图片的内存占用大致的计算公式如下:

占用内存 = 图像像素总和(width x height)再 x 每像素(bitmap config)占用的字节数

以下是通过代码准确计算

    public static int getSizeInBytes(@Nullable Bitmap bitmap) {
        if (bitmap == null) {
            return 0;
        }

        // There's a known issue in KitKat where getAllocationByteCount() can throw an NPE. This was
        // apparently fixed in MR1: http://bit.ly/1IvdRpd. So we do a version check here, and
        // catch any potential NPEs just to be safe.
        if (Build.VERSION.SDK_INT > Build.VERSION_CODES.KITKAT) {
            try {
                return bitmap.getAllocationByteCount();
            } catch (NullPointerException npe) {
                // Swallow exception and try fallbacks.
            }
        }

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB_MR1) {
            return bitmap.getByteCount();
        }

        // Estimate for earlier platforms.
        return bitmap.getWidth() * bitmap.getRowBytes();
    }

以一张10241024的图片为例,使用ARGB_8888,占用的内存为10241024*4=4M。像现在的手机摄像头动不动就是上千万像素,拍出来的照片如果按默认的ARGB_8888 config加载,则至少是几十M的内存占用。

Android的图片资源主要分两部分:

  1. 一种是apk中自带的,多为png格式,由系统加载,支持缩放,代码中通过R.xxx引用,decode时使用的是默认的ARGB_8888选项,图像质量高;
  2. 另一种是网络图片或本地图片,多为jpg格式,加载时一般使用第三方的图片加载库,为节省内存decode时多为RGB_565选项。

平时都是这么用,也没发现问题,优化内存占用时,一般也是从图片的尺寸方面入手。不过最近优化一个跟图片相关的功能,在图片尺寸无法缩放的条件下,只能通过更改bitmap config来降低内存的占用。然后意外的发现,导致颠覆了我的三观。为此我特地写了一个测试sample,代码详见github,特地创建了一张背景色透明,图片内容为A(黑色50%透明度)R(红色)G(绿色)B(蓝色)图片,然后分别导出为:png8(alpha透明)、png24(不透明)、png32和jpeg(不透明)格式的图,分别使用ALPHA_8, ARGB_4444, ARGB_8888, RGB_565四种config加载图片,得到的实际结果如下(假设图像总像素为X)。

实践结果

运行截图

screenshotscreenshot

结果统计

bitmap.config ALPHA_8 ARGB_4444 ARGB_8888 RGB_565
png8 41 X 2 X 4 X 42 X A 透明
png24 41 X 2 X 4 X 2 X
png32 41 X 2 X 4 X 42 X A 透明
jpeg 41 X 2 X 4 X 2 X

请注意表格中带删除线的部分

  1. ALPHA_8:config占用的内存竟然和ARGB_8888一样,不是说每个像素占用1字节的么?
  2. RGB_565:在png8和png32中,图片中的A都保持了50%的透明度,而且占用的内存也和ARGB_8888一样,不是说RGB_565不包含alpha么?不是说占用的内存是ARGB_8888的一半么?
  3. ARGB_4444:在android 6.0上面,png8和png32看不见(全透明),png24和jpeg显示为一块黑色区域,在android 4.2上则显示正常。

带着上面的疑问,在网上进行了相关的搜索,也没有找到答案。好吧,我是懵了,不知道各位看客如何?附上github上的示例工程:https://github.com/Jamling/BitmapConfig

本文永久链接: http://www.ieclipse.cn/2017/06/14/Android/Android-bitmap-config/ 未经允许,禁止转载,如有问题,请在我的博客原始页面提交评论。

上一篇下一篇

猜你喜欢

热点阅读