Bitmap处理

开发中避免不了要与Bitmap打交道,但随着手机的屏幕及分辨率提高图片质量也得到很大提高,这就当来一系列的BUG源。
一个就是OOM,OOM不单给程序带来问题,还对用户的使用带来不好的体验(这里就不能容忍了)。

Bitmap怎么会引起OOM的?

  • 1.每个机型在编译的时候ROM都设置了一个应用对内存VM的上限,用来限制每个应用可用的最大内存,当程序的运行内存超过这个阈值就会内存溢出OOM。一般根据手机屏幕dpi大小递增。
  • 2.图片分辨率越高,消耗的内存越大,当加载高分辨率图片的时候,将会非常占用内存,一旦处理不当就会OOM。
  • 3.在使用ListView, GridView等这些大量加载view的组件时,如果没有合理的处理缓存,大量加载Bitmap的时候,也将容易引发OOM。

Bitmap所占内存

Bitmap所占用的内存 = 图片长度 x 图片宽度 x 一个像素点占用的字节数。
而Bitmap.Config,正是指定单位像素占用的字节数的重要参数。

ALPHA_8 
表示8位Alpha位图,即A=8,一个像素点占用1个字节,它没有颜色,只有透明度 。
ARGB_4444 
表示16位ARGB位图,即A=4,R=4,G=4,B=4,一个像素点占4+4+4+4=16位,2个字节 。
ARGB_8888 
表示32位ARGB位图,即A=8,R=8,G=8,B=8,一个像素点占8+8+8+8=32位,4个字节 。
RGB_565 
表示16位RGB位图,即R=5,G=6,B=5,它没有透明度,一个像素点占5+6+5=16位,2个字节。
注:A代表透明度;R代表红色;G代表绿色;B代表蓝色。

Bitmap加载

BitmapFactory提供的解析Bitmap的静态工厂方法有以下五种:
decodeFile(...)
decodeResource(...)
decodeByteArray(...)
decodeStream(...)
decodeFileDescriptor(...)
其中常用的三个:decodeFile、decodeResource、decodeStream。
decodeFile和decodeResource其实最终都是调用decodeStream方法来解析Bitmap。

Bitmap优化

1.BitmapConfig的配置
2.使用decodeFile、decodeResource、decodeStream进行解析Bitmap时,配置inDensity和inTargetDensity,两者应该相等,值可以等于屏幕像素密度*0.75f
3.使用inJustDecodeBounds预判断Bitmap的大小及使用inSampleSize进行压缩
4.对Density>240的设备进行Bitmap的适配(缩放Density)
5.4.4以下版本inPurgeable、inInputShareable的使用
7.Bitmap的回收
所以我们根据以上的思路,我们将Bitmap优化的策略总结为以下3种:
    1.对图片质量进行压缩
    2.对图片尺寸进行压缩
    3.使用libjpeg.so库进行压缩

对图片质量进行压缩

    ByteArrayOutputStream baos = new ByteArrayOutputStream();      
    //质量压缩方法,这里100表示不压缩,把压缩后的数据存放到baos中      
    //一般是循环判断如果压缩后图片是否大于设置的KB      
    bitmap.compress(Bitmap.CompressFormat.JPEG, 100, baos); 

    //把压缩后的数据baos存放到ByteArrayInputStream中      
    ByteArrayInputStream isBm = new ByteArrayInputStream(baos.toByteArray());      
    //把ByteArrayInputStream数据生成图片      
    Bitmap newBitmap = BitmapFactory.decodeStream(isBm, null, null);      

对图片尺寸进行压缩

/** 根据手机屏幕的宽高设置图片的大小 */
    public Bitmap getFitBitmap(InputStream inputStream ,int mWidth ,int mHeight) {
        if (inputStream == null) {
            return null;
        }
        BitmapFactory.Options options = new BitmapFactory.Options();
        options.inJustDecodeBounds = true;
        try {
            BitmapFactory.decodeStream(inputStream ,new Rect(0 ,0 ,0 ,0) ,options);

            int width = options.outWidth;
            int height = options.outHeight;

            int sampleSize = 1;

            if (width > mWidth || height >mHeight) {
                while (width/sampleSize > mWidth || height/sampleSize >mHeight) {
                    sampleSize *= 2;
                }
            }
            options = new BitmapFactory.Options();
            options.inJustDecodeBounds = false;
            options.inSampleSize = sampleSize;
            options.inPreferredConfig = Bitmap.Config.RGB_565;

            return BitmapFactory.decodeStream(inputStream ,new Rect(0 ,0 ,0 ,0) ,options);
        }catch (Exception e) {
            Log.e("BitmapUtil" ,"=== "+ e);
        }
        return null;
    }

使用libjpeg.so库进行压缩

除了通过设置simpleSize根据图片尺寸压缩图片和通过Bitmap.compress方法通过压缩图片质量两种方法外,我们还可以使用libjpeg.so这个库来进行压缩。

libjpeg是广泛使用的开源JPEG图像库,Android所用的是skia的压缩算法,而Skia对libjpeg进行了的封装。
libjpeg在压缩图像时,有一个参数叫optimize_coding,关于这个参数,libjpeg.doc有如下解释:

boolean optimize_coding TRUE causes the compressor to compute optimal
Huffman coding tables for the image. This requires an extra pass over the
data and therefore costs a good deal of space and time. The default is
FALSE, which tells the compressor to use the supplied or default Huffman
tables. In most cases optimal tables save only a few percent of file size
compared to the default tables. Note that when this is TRUE, you need not
supply Huffman tables at all, and any you do supply will be overwritten.
如果设置optimize_coding为TRUE,将会使得压缩图像过程中基于图像数据计算哈弗曼表,由于这个计算会显著消耗空间和时间,默认值被设置为FALSE。

谷歌的Skia项目工程师们最终没有设置这个参数,optimize_coding在Skia中默认的等于了FALSE,但是问题就随之出现了,如果我们想在FALSE和TRUE时压缩成相同大小的JPEG 图片,FALSE的品质将大大逊色于TRUE的,尽管谷歌工程师没有将该值设置为true,但是我们可以自己编译libjpeg进行图片的压缩。

libjpeg的官网下载地址:http://www.ijg.org/ 。从官网下载之后,我们必须自己对其进行编译。

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

推荐阅读更多精彩内容