Android | Bitmap解析

Android中Bitmap是对图像的一种抽象。通过他可以对相应的图像进行剪裁,旋转,压缩,缩放等操作。这里循序渐进的一步步了解Bitmap的相关内容。

先了解Bitmap相关的API,然后根据API进一步了解内部的实现。


1.生成Bitmap--BitmapFactory

android.graphics.Bitmap.java

 /**
* Private constructor that must received an already allocated native bitmap
* int (pointer).
*/
// called from JNI Bitmap类中只有一个构造器。而且还不能直接调,必须通过JNI去构造
Bitmap(long nativeBitmap, byte[] buffer, int width, int height, int density,
boolean isMutable, boolean requestPremultiplied,
byte[] ninePatchChunk, NinePatch.InsetStruct ninePatchInsets) {
...
}

Bitmap类中只有一个构造器。而且还不能直接调,必须通过JNI去构造。源码中已经写的很清楚。所以一定有JNI的方式产生Bitmap对象。这就是BitmapFactory的功能--生成Bitmap

android.graphics.BitmapFactory.java

	/**
* Creates Bitmap objects from various sources, including files, streams,
* and byte-arrays.
*/
public class BitmapFactory {
...
}

通过这个类可以看到所有的方法都是静态方法。而且支持从文件,流,字节数组等各种资源中获取Bitmap对象。

从结构图可以看出生成Bitmap的方式有四大类:

decodeFile 从文件系统加载

decodeResource 以R.drawable.xxx的形式从本地资源中加载

decodeStream 从输入流加载

decodeByteArray 从字节数组中加载

其中decodeFile和decodeResource间接调用decodeStream方法。而且返回Bitmap使用前要判空,在不存在的情况下会返回null.

每一种decodeXXX方法最后都有一个Options对象。这个对象的作用就是在decode解压到内存之前的一些可选项。利用这些可选项可以优化内存,以及展示的效果等。

Options是BitmapFactory的静态内部类

常用的配置有:

inPreferredConfig | 指定位图编码格式 可选项在Bitmap.Config中,缺省值是ARGB_8888。

inJustDecodeBounds | true 返回的Bitmap将是null,但是Options的outAbc中解出了图像的基本信息

inSampleSize | 采样率,设置缩放比例.采样率就是抽取像素点组成位图,采样率同时作用于宽和高。

比如inSampleSize为2,采样后的图片的宽高均为原始图片宽高的1/2,这时像素=原始图片的1/4,占用内存=原始图片的1/4;

inSampleSize的取值应该总为2的整数倍,否则会向下取整,取一个最接近2的整数倍,比如inSampleSize=3时,系统会取inSampleSize=2

假设一张10241024,模式为ARGB_8888的图片,inSampleSize=2,原始占用内存大小是4MB,采样后的图片占用内存大小就是(1024/2) * (1024/2 ) 4 = 1MB

备注:

ALPHA_8:每个像素占用1byte内存。

ARGB_4444:每个像素占用2byte内存

ARGB_8888:每个像素占用4byte内存(默认为改格式)

RGB_565:每个像素占用2byte内存

所以默认一张1024*1024像素的位图是4MB的,如果硬生生加载进内存会占用很大一块。很容易造成OOM!因此就需要Options去提前配置一下。


2.Bitmap方法对自身的变换

Bitmap可以和Matrix结合实现图像的剪切、旋转、缩放等操作,最终调用到的createXXXBitmapXXX方法有下面几个:

他们最终都会调用native方法:

//////////// native methods

private static native Bitmap nativeCreate(int[] colors, int offset,
int stride, int width, int height,
int nativeConfig, boolean mutable);

可见他们的变换本质都是一样的。只是调用的方式不同

用源Bitmap通过变换生成新的Bitmap的方法:

public static Bitmap createBitmap(Bitmap source, int x, int y, intwidth, int height,
Matrix m, boolean filter) //最终的实现,后两种只是对第一种方法的封 ,
//设置Matrix的Rotate(通过setRotate())或者Scale(通过setScale()),传入第一个方法,可实现旋转或缩放。 public static Bitmap createBitmap(Bitmap source, int x, int y, intwidth, int height)
//可以从源Bitmap中指定区域(x,y, width, height)中挖出一块来实现剪切 public static Bitmap createScaledBitmap(Bitmap src, int dstWidth,
int dstHeight,boolean filter)
//可以把源Bitmap缩放为dstWidth x dstHeight的Bitmap

用于压缩到流的方法:

public boolean compress(CompressFormat format, int quality, OutputStream stream)
//format指定压缩的格式:
一共三种
JPEG (0),
PNG (1),
WEBP (2);
//quality指定压缩品质0-100可输入。0是最差,100是无损。PNG格式压缩会忽略这个选项
//stream 压缩输出的流在这里取出,接下去可以对流进行操作 public Bitmap copy(Config config, boolean isMutable)
//这个方法主要用于复制出一个全新的位图。Config指定新位图格式ALPHA_8, RGB_565, ARGB_4444, ARGB_8888,
//isMutable, true表示产生的位图无法更改

3.使用Bitmap需要注意的问题

1.最常见的问题OOM

及时释放内存

if (!bmp.isRecycle()) {
bmp.recycle(); //回收图片所占的内存
bitmap = null;
system.gc(); //提醒系统及时回收
}

recycle()并不保证立即释放占用的内存,但是可以加速内存的释放。

释放内存以后,就不能再使用该Bitmap对象了,否则会抛异常。所以一定要保证不再使用再释放。比如,如果是在某个Activity中使用Bitmap,建议在Activity的onStop()或者onDestroy()中进行回收。

内存中混存通用Bitmap避免多次加载。

比如默认图片可以只加载一次然后一个界面内多处使用。

2.图片压缩

inSampleSize压缩是尺寸压缩,compress方法是质量压缩。

compress方法在保持像素的前提下改变图片的位深及透明度等,来达到压缩图片的目的,经过它压缩的图片文件存储大小会改变,但导入成bitmap后占内存大小不变,宽高也不会改变。

Android笔记--Bitmap的更多相关文章

  1. Android笔记——Bitmap自动取色(纯搬运)

    2015/6/12更新:发现一个更好的,带demo https://github.com/MichaelEvans/ColorArt 说明: 这个是一个老外写的自动自动从bitmap中取主色与第二主色 ...

  2. Android笔记--Bitmap(三) 针对不用Android版本的位图管理

    Bitmap(三) | Android不同版本的相应操作 在不同的Android版本中.位图的存储方式是不同的. 1.小于等于 Android 2.2 (API level 8) 垃圾收集器回收内存时 ...

  3. Android笔记--Bitmap(二)内存管理

    Bitmap(二) 内存管理 1.使用内存缓存保证流畅性 这种使用方式在ListView等这种滚动条的展示方式中使用最为广泛, 使用内存缓存 内存缓存位图可以提供最快的展示.但代价就是占用一定的内存空 ...

  4. Android笔记--自定义控件仿遥控器的圆形上下左右OK圆盘按钮

    原文:Android笔记--自定义控件仿遥控器的圆形上下左右OK圆盘按钮 上面就是几张预览图!代码在最底下 主要就两个步骤,画图.监听点击 1.整个控件基本上是一步步画出来的,重写onDraw方法开始 ...

  5. Android笔记——Matrix

    转自:http://www.cnblogs.com/qiengo/archive/2012/06/30/2570874.html#translate Matrix的数学原理 在Android中,如果你 ...

  6. Android笔记——Android自定义控件

    目录: 1.自定义控件概述 01_什么是自定义控件 Android系统中,继承Android系统自带的View或者ViewGroup控件或者系统自带的控件,并在这基础上增加或者重新组合成我们想要的效果 ...

  7. int android.graphics.Bitmap.getRowBytes()

    int android.graphics.Bitmap.getRowBytes() Return the number of bytes between rows in the bitmap's pi ...

  8. Android笔记——Android中数据的存储方式(二)

    我们在实际开发中,有的时候需要储存或者备份比较复杂的数据.这些数据的特点是,内容多.结构大,比如短信备份等.我们知道SharedPreferences和Files(文本文件)储存这种数据会非常的没有效 ...

  9. Android中Bitmap, Drawable, Byte,ID之间的转化

    Android中Bitmap, Drawable, Byte,ID之间的转化 1.  Bitmap 转化为 byte ByteArrayOutputStream out = new ByteArray ...

随机推荐

  1. webpack 知识点总结

    1 webpack 的不同点: 1.1 实现代码分割 1.2 loaders 1.3 plugin system 2 webpack 打包css,需要使用css-loaders 使webpack识别c ...

  2. bzoj4520

    KD-tree+堆 多年大坑 KD-tree已经是半年前学的了,忘记了.这道题当时一直T,今天重新抄了一遍,A了 KD-tree过程:1.建树:每次依次按x,y划分平面,像二叉搜索树一样建树,每个点维 ...

  3. 转 vs2008使用技巧推荐

    Visual Studio 2008自带的1000多个 Windows 系统使用的各种图标.光标和动画文件在Visual Studio 2008的安装目录下,\Microsoft Visual Stu ...

  4. CPython里的GIL

    GIL不是Python特性,是CPython解释器特性,因为CPython有垃圾回收机制. GIL 本质是互斥锁,保护解释器安全. 保证线程安全,垃圾回收线程不会和其他线程一起运行. 多个线程不能实现 ...

  5. Flutter实战视频-移动电商-06.Dio基础_Get请求和动态组件协作

    博客地址: https://jspang.com/post/FlutterShop.html#toc-0ee 编写页面代码 创建动态组件HomePage,原来的代码是静态的我们这里就去掉就可以了. 然 ...

  6. Linux 静态库 & 动态库

    转自:http://blog.chinaunix.net/uid-26833883-id-3219335.html 一.什么是库   本质上来说库是一种可执行代码的二进制形式,可以被操作系统载入内存执 ...

  7. POJ - 2376 Cleaning Shifts 贪心(最小区间覆盖)

    Cleaning Shifts Farmer John is assigning some of his N (1 <= N <= 25,000) cows to do some clea ...

  8. Unity AnimatorController注意事项

    通过assetbundle加载的单独打包AnimatorController使用下面方法赋值 Go.GetComponent<Animator>().runtimeAnimatorCont ...

  9. python+smtplib 发送测试报告到邮箱

    之前有介绍过怎样快速的搭建一个python测试框架 python+unittest 搭建简易的接口测试框架 这里介绍一下,怎样使用smtplib将测试报告发送到邮箱,这样使用jenkins定时巡检,执 ...

  10. IQueryable 和IEnumberable的区别

    一.IEnumerable接口 公开枚举器,该枚举器支持在指定类型的集合上进行简单的迭代.即:实现了此接口的object,就可以使用foreach遍历该object: 二.IQueryable 接口 ...