为了避免oom的出现,几乎每个应用都会对大图进行压缩,我现在手头做的产品就有很多地方用到,以前封装工具类的时候,都是在网上找东找西,然后拼拼凑凑,有效果就行了,一直用的迷迷糊糊,这几天工作比较闲,正好系统的总结梳理一下图片压缩方式:
图片压缩现在常见的有三种方式:
1、等比压缩,等比压缩是保持原图长宽比例的压缩,只是图片变小,展示的还是原图的所有内容(区别于第二种通过Matrix压缩,可以选取图片的一部分,类似于上传头像时,让你在图上选一块zoom的形式)。等比压缩用的的主要是BitmapFactory.Options,通过options缩放比例的设置,来生成缩略图:
/**
* @param path 图片路径
* @param targetSize 缩放后期待的长边(图片长和宽大的那一个边)的长度
* @param targetW 期待的缩放后宽度
* @param targetH 期待的缩放后高度
* @return
*/
public static Bitmap equalRatioScale(String path,int targetW,int targetH){
// 获取option
BitmapFactory.Options options = new BitmapFactory.Options();
// inJustDecodeBounds设置为true,这样使用该option decode出来的Bitmap是null,
// 只是把长宽存放到option中
options.inJustDecodeBounds = true;
// 此时bitmap为null
Bitmap bitmap = BitmapFactory.decodeFile(path, options);
int inSampleSize = 1; // 1是不缩放
// 计算宽高缩放比例
int inSampleSizeW = options.outWidth / targetW;
int inSampleSizeH = options.outHeight / targetH;
// 最终取大的那个为缩放比例,这样才能适配,例如宽缩放3倍才能适配屏幕,而
// 高不缩放就可以,那样的话如果按高缩放,宽在屏幕内就显示不下了
if (inSampleSizeW > inSampleSizeH) {
inSampleSize = inSampleSizeW;
}else {
inSampleSize = inSampleSizeH;
}
// 设置缩放比例
options.inSampleSize = inSampleSize;
// 一定要记得将inJustDecodeBounds设为false,否则Bitmap为null
options.inJustDecodeBounds = false;
bitmap = BitmapFactory.decodeFile(path, options);
return bitmap;
}
2、通过Matrix进行更加灵活的缩放:这种方式主要是通过构建缩放矩阵和Bitmap.createBitmap方法来实现灵活缩放,宽和高缩放的比例可以不一致,而且通过Bitmap.createBitmap方法创建出来的是新的位图,这个位图可以是选取原图的一部分,而不是对原图进行整体缩放!类似于上传头像时,让你在原图上扣下来一块的效果,控制非常灵活。
/**
* @param path 原图路径
* @param offsetX 截取开始点在X轴偏移量
* @param offsetY 截取开始点在Y轴偏移量
* @param targetW 截取多宽(像素)
* @param targetH 截取多高(像素)
* @return
*/
public static Bitmap matrixScale(String path,int offsetX,int offsetY,int targetW,int targetH){
// 构建原始位图
Bitmap bitmap = BitmapFactory.decodeFile(path);
// 获取原始宽高
int width = bitmap.getWidth();
int height = bitmap.getHeight();
// 计算宽高缩放比例,targetW,targetH即期待缩放完成后位图的宽高
float scaleW = (float)targetW / width;
float scaleH = (float)targetH / height;
// 将缩放比例放进矩阵
Matrix matrix = new Matrix();
matrix.postScale(scaleW, scaleH);
// 这个方法作用非常多,详细解释一下各个参数的意义!
// bitmap:原始位图
// 第二到第五个参数,即截取原图哪一部分构建新位图,
// offsetX和offsetY代表在X轴和Y轴上的像素偏移量,即从哪个位置开始截取
// width和height代表截取多少个像素,但是要注意,offsetX+width应该小于等于原图的宽度
// offsetY+height小于等于原图高度,要不然会报错,因为截到原图外面去了
// 像下面这样填写,就代表截取整个原图,
// Bitmap.createBitmap(bitmap, 0, 0, width, height, matrix, false);
// 如果填写100,100,200,200,就代表
// 从原图左上角往右和下各偏移100像素,然后往后和往下各截取200构建新位图
// matrix:缩放矩阵
// 最后一个参数表示如果矩阵里面还存放了过滤条件,是否按条件过滤(如果matrix里面只放了平移数据),最后一个参数设置成什么都不会生效
bitmap = Bitmap.createBitmap(bitmap, offsetX, offsetY, width, height, matrix, false);
return bitmap;
}
3、无损压缩,无损压缩是说图片大小和清晰度看上去和原图没有什么差别,但是确实size缩小了,这里缩小的原理是牺牲了分辨率等其他直观看不到的东西,看起来和原图一样,但是一放大就立马失真了,不会和原图一样放大很多才会逐渐变得不清晰。无损压缩后的图片像素并不会减少,而Bitmap占用内存的定义就是像素点占的内存,所以以Bitmap的方式加载到内存中时,和压缩前占用的内存是同样大的,原来会oom的图片,质量压缩后同样会oom;但是,质量压缩后将流输出到文件中,文件的size会大幅度减小,所以质量压缩特别适合在Android端进行图片上传的时候进行图片压缩,既能保持上传后的清晰度,又能减小size。另外质量压缩不是可以无限缩小的,降低到一定程度,就算把quality设置的再小,size也不会再降低了。另外这种方式最好返回保存压缩后的图片保存的文件路径,而不要直接返回Bitmap,示例就懒得改了。
<pre name="code" class="java">/**
* @param path 图片路径
* @param quality 质量 0-100,100表示原图
* @return
*/
public static Bitmap losslessScale(String path,int quality){
Bitmap bitmap = BitmapFactory.decodeFile(path);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
bitmap.compress(CompressFormat.JPEG, quality, baos);
Log.e("哈哈","原始大小:" + baos.toByteArray().length);
// 因为质量压缩不是可以无限缩小的,所以一张高质量的图片,再怎么压缩,
// 最终size可能还是大于你指定的size,造成异常
// 所以不建议循环压缩,而是指定quality,进行一次压缩就可以了
// while (baos.toByteArray().length / 1024 > maxSize) {
// quality -= 10;
// baos.reset();
// bitmap.compress(CompressFormat.JPEG, quality, baos);
// Log.e("哈哈","过程中大小为:"
// + baos.toByteArray().length);
// }
bitmap.compress(CompressFormat.JPEG, quality, baos);
Log.e("哈哈","最终大小" + baos.toByteArray().length);
Bitmap compressedBitmap = BitmapFactory.decodeByteArray(
baos.toByteArray(), 0, baos.toByteArray().length);
return compressedBitmap;
}
一张图片处理过程,建议先进行等比压缩或者Matrix压缩后,再进行质量压缩,这样组合使用,不管是生成缩略图还是图片上传,效果都不错