Bitmap系列-Options

这是我的第一篇关于Bitmap的文章,如有理解错误之处,烦请大家能及时留言提出,我会及时纠正,谢谢。

今天主要和大家探讨的是BitmapFactory.Options的使用.

目录:

第一部分:BitmapFactory.Options属性的理解

第二部分:从源码角度查看Options是如何工作的


BitmapFactory.Options

接下来分析一下Options中一些重要属性.

先放类图


BitmapFactory.Options

inDither(***)

1:定义:是否对位图抖动处理

2:作用:会让图片的显示效果更平滑一些,提高显现效果.

3:使用场景:颜色模式高的图片(例如ARGB_8888)采用低颜色模式(例如:RGB_565)去解码的时候,需要把此值设置为true.

4:原理:https://blog.csdn.net/haozipi/article/details/47185535(这个说的比较清楚)

inPreferQualityOverSpeed(**)

1:API解析:如果inPreferQualityOverSpeed设置为true,则解码器将尝试将重建图像解码为更高质量,即使牺牲解码速度.目前该领域只影响JPEG解码,在这种情况下,将使用更精确但速度稍慢的IDCT方法.

inPremultiplied(***)

1:API解释:如果为true(这是默认值),则生成的位图的颜色通道将由Alpha通道预先倍增。

对于由视图系统或Canvas直接绘制的图像,这不应设置为false。 视图系统和Canvas假定所有绘制的图像都被预乘以简化绘制时间混合,并且在绘制未预乘积时会抛出RuntimeException。

这可能仅在您想操纵原始编码图像数据时才有用,例如, 使用RenderScript或自定义OpenGL。

这不会影响没有Alpha通道的位图。

将inScaled设置为true时将此标志设置为false可能会导致不正确的颜色。

2:注意:inScaled设置为true,inPremultiplied设置为false,并且图像具有alpha通道(例如加载png图片),则会出错"java.lang.RuntimeException: Canvas: trying to use a non-premultiplied bitmap"

inPreferredConfig(*****)

1:定义:解码图片生成Bitmap时的颜色模式.

2:作用:标记每个像素点占的内存大小(单位是字节)

3:疑问:目前从api说明来看,不同的Config对应的每个像素点占的内存是不一样的,但是测试时发现,ALPHA_8,RGB_565,ARGB_8888占用的内存一样大,目前还不知道原因??

inSampleSize(*****)

1:采样率

2:取值:2的整数幂(小于1则取值为1,任何其他值将向下舍入到最接近的2的幂)

3:作用:对图片进行二次采样,使Bitmap占用的内存大小降低,使Bitmap的宽和高减小(宽高变为原图的1/sampleSize,进而Bitmap占用的内存变小为1/sampleSize^2)

4:场景:大图加载,缩略图

inJustDecodeBounds(*****)

1:是否仅仅测量Bitmap的边界

2:取值:true,只测量Bitmap的宽高等属性,并且解码返回的Bitmap为null. 这些属性值从outWitdh,outHeight,outMimeType(图片类型)获取.

inMutable(*****Mutable:易变的,可变的)

1:作用:标记位图是否可变,是否可以对该位图做修改.

2:取值:true:可变,false:不可变

3:注意点:如果用Options加载的Bitmap的话,那么此值对应的是Bitmap的mIsMutable属性.

如果是不可变的话,例如调用setPixel()修改位图中像素点的颜色会抛出"IllegalStateException(非法状态异常)"

inBitmap(*****)

1:作用:指示解码器重复使用已经加载的Bitmap的内存空间去加载新位图从而达到节省内存的目的.

2:注意点:1:重用位图的mIsMutable的值为false,那么新加载Bitmap就不能用重用位图的内存空间解码图片。此时新加载的Bitmap与重用的Bitmap不是同一个。

2:重用位图的mIsMutable的值为true,如果要加载位图的getByteCount()字节数大于重用位图的Bitmap.getAllocationByteCount()分配的字节数,

那么调用BitmapFactory.decode()会返回 null 并将抛出IllegalArgumentException。

3:重用位图的mIsMutable的值为true,如果要加载位图的getByteCount()字节数小于或等于重用位图的Bitmap.getAllocationByteCount()分配的字节数,

BitmapFactory就可以重复使用可变位图来解码任何其他位图.

此时:1:重用位图与新加在位图是同一个对象

2:重用位图的尺寸变成了新加载位图的尺寸

3:重用位图的内存大小与新加载位图的位图内存大小一样,且都为重用位图的大小

4:重用位图的内容变成了新加载位图的内容(从源码来看也确实是这样,因为重用位图像素信息存储的是新加载位图的像素信息)

inScaled(*****)

1:定义:是否缩放,而API解释则更清楚一些:设置此标志时,如果inDensity和inTargetDensity不为0,则位图将在加载时缩放以匹配inTargetDensity,而不是每次绘图到Canvas时都依靠图形系统进行缩放。

BitmapRegionDecoder将忽略此标志,并且不会根据密度缩放输出。 (尽管支持inSampleSize)默认情况下,此标志处于打开状态,如果需要位图的非缩放版本,应将其关闭。 九个补丁位图忽略这个标志并且总是缩放。

如果inPremultiplied设置为false,并且图像具有alpha,则将此标志设置为true可能会导致不正确的颜色。

2:作用:缩放Bitmap

inDensity(*****)

1:定义:该位图是在哪种像素密度下创建的(这个理解的可能不对,如果理解不对的话,希望大神们多多指出).

2:取值:DisplayMetrics中的值.

3:作用:与inScaled,inTargetDensity,inScreenDensity一起使用,决定最终Bitmap的宽高

4:note:当Options.inDensity不为0的话则把该值设置到Bitmap的mDensity(see BitmapFactory.setDensityFromOptions()).

inTargetDensity(*****)

1:定义:该位图要加载并显示到的目标设备上的像素密度。

2:取值:DisplayMetrics中的值.

3:作用:与inScaled,inTargetDensity,inScreenDensity一起使用,决定最终Bitmap的宽高

4:note:当Options.inScaled或isNinePatch一项为真则把该值设置到Bitmap的mDensity(see BitmapFactory.setDensityFromOptions()).

inScreenDensity(*****)

1:定义:这个不知道怎么理解,希望打什么能指出

2:取值:DisplayMetrics中的值.

3:作用:1:与inScaled,inTargetDensity,inScreenDensity一起使用,决定最终Bitmap的宽高.

          2:通过与inDensity比较是否相等来确认是否需要缩放位图(相等不缩放,不相等则缩放)

inPurgeable与inInputShareable

inPurgeable与inInputShareable联合使用,从Build.VERSION_CODES.LOLLIPOP(21)开始,可以采用inBitmap替换.

网上找的解释:inPurgeable与inInputShareable 二个是并列使用,如果设置了inPurgeable = false,则inInputShareable的值会被忽略;

这二个选项的作用主要是便于系统及时回收bitmap占用的内存;

inPurgeable:设置为True,则使用BitmapFactory创建的Bitmap用于存储Pixel的内存空间,在系统内存不足时可以被回收,当应用需要再次访问

该Bitmap的Pixel时,系统会再次调用BitmapFactory 的decode方法重新生成Bitmap的Pixel数组。 设置为False时,表示不能被回收。

inInputShareable:设置是否深拷贝,与inPurgeable结合使用,inPurgeable为false时,该参数无意义.


从源码层分析Options是如何工作的

下面简单从BitmapFactory.cpp中分析各属性的作用(分析的很简单,几乎每个开发同学拿到这个源码都能看懂,所以就稍微梳理下加深下印象就行了)。

•inSampleSize:这里不做深究底层是如何实现的,有兴趣的小伙伴可以自行研究下

•inJustDecodeBounds:

•inSacled与inDensity与nTargetDensity与inScreenDensity

这几个属性共同决定了最终要显示的Bitmap的宽/高。

•inBitmap:

好了,今天的分享就到这里了,有什么理解错误的地方,欢迎大家积极指出,我会尽快改正的,谢谢。

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

推荐阅读更多精彩内容