android 图片压缩

下面先讲解关于BitmapFactory.options选项的所有字段:

android 图片压缩_第1张图片

以上是摘自Android官方的文档:

下面我们说一个问题:

怎么获取图片的大小呢?

首先我们要把这个图片转化成Bitmap,然后在利用Bitmap的getwidth()和getHeight()方法就可以取得图片的宽和高了,

但是此时问题来了,在通过BitmapFactory.decodeFile(Sting file)方法转化成Bitmap时,遇到大一些的图片时,我们经常遇到OOM(out of memory)的问题,怎么避免呢?

此时就用到了上面提到的BitmapFactory.options这个类:

BitmapFactory.options这个类,有一个字段为: inJustDecodeBounds

如果我们把它设为true,那么BitmapFactory.decodeFile(String path, Options opt)并不会真的返回一个Bitmap给你,它仅仅会把它的宽,高取回来给你,这样就不会占用太多的内存,也就不会那么频繁的发生OOM了。

BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
Bitmap bmp = BitmapFactory.decodeFile(path, options);
/* 这里返回的bmp是null */

这段代码之后,options.outWidth 和 options.outHeight就是我们想要的宽和高了。

有了宽,高的信息,我们怎样在图片不变形的情况下获取到图片指定大小的缩略图呢?
比如我们需要在图片不变形的前提下得到宽度为200的缩略图。
那么我们需要先计算一下缩放之后,图片的高度是多少 
/* 计算得到图片的高度 */
/* 这里需要主意,如果你需要更高的精度来保证图片不变形的话,需要自己进行一下数学运算 */
int height = options.outHeight * 200 / options.outWidth;
options.outWidth = 200;
options.outHeight = height; 
/* 这样才能真正的返回一个Bitmap给你 */
options.inJustDecodeBounds = false;
Bitmap bmp = BitmapFactory.decodeFile(path, options);
image.setImageBitmap(bmp);
这样虽然我们可以得到我们期望大小的ImageView
但是在执行BitmapFactory.decodeFile(path, options);时,并没有节约内存。要想节约内存,还需要用到BitmapFactory.Options这个类里的 inSampleSize 这个成员变量。
我们可以根据图片实际的宽高和我们期望的宽高来计算得到这个值。
inSampleSize = options.outWidth / 200;
另外,为了节约内存我们还可以使用下面的几个字段:
options.inPreferredConfig = Bitmap.Config.ARGB_4444;    // 默认是Bitmap.Config.ARGB_8888
/* 下面两个字段需要组合使用 */
options.inPurgeable = true;
options.inInputShareable = true;


===========

[java]  view plain  copy
  1. 1、设置缩放大小对图片作处理  
  2.   
  3. public Bitmap getBitmapFromFile(File dst, int width, int height) {  
  4.     if (null != dst && dst.exists()) {  
  5.         BitmapFactory.Options opts = null;  
  6.         if (width > 0 && height > 0) {  
  7.             opts = new BitmapFactory.Options();  
  8.             opts.inJustDecodeBounds = true;  
  9.             BitmapFactory.decodeFile(dst.getPath(), opts);  
  10.             // 计算图片缩放比例  
  11.             final int minSideLength = Math.min(width, height);  
  12.             opts.inSampleSize = computeSampleSize(opts, minSideLength,  
  13.                     width * height);  
  14.             opts.inJustDecodeBounds = false;  
  15.             opts.inInputShareable = true;  
  16.             opts.inPurgeable = true;  
  17.         }  
  18.         try {  
  19.             return BitmapFactory.decodeFile(dst.getPath(), opts);  
  20.         } catch (OutOfMemoryError e) {  
  21.             e.printStackTrace();  
  22.         }  
  23.     }  
  24.     return null;  
  25. }  
  26.    
  27.   
  28.   
  29.    
  30.   
  31. public static int computeSampleSize(BitmapFactory.Options options,  
  32.         int minSideLength, int maxNumOfPixels) {  
  33.     int initialSize = computeInitialSampleSize(options, minSideLength,  
  34.             maxNumOfPixels);  
  35.   
  36.     int roundedSize;  
  37.     if (initialSize <= 8) {  
  38.         roundedSize = 1;  
  39.         while (roundedSize < initialSize) {  
  40.             roundedSize <<= 1;  
  41.         }  
  42.     } else {  
  43.         roundedSize = (initialSize + 7) / 8 * 8;  
  44.     }  
  45.   
  46.     return roundedSize;  
  47. }  
  48.   
  49. private static int computeInitialSampleSize(BitmapFactory.Options options,  
  50.         int minSideLength, int maxNumOfPixels) {  
  51.     double w = options.outWidth;  
  52.     double h = options.outHeight;  
  53.   
  54.     int lowerBound = (maxNumOfPixels == -1) ? 1 : (int) Math.ceil(Math  
  55.             .sqrt(w * h / maxNumOfPixels));  
  56.     int upperBound = (minSideLength == -1) ? 128 : (int) Math.min(Math  
  57.             .floor(w / minSideLength), Math.floor(h / minSideLength));  
  58.   
  59.     if (upperBound < lowerBound) {  
  60.         // return the larger one when there is no overlapping zone.  
  61.         return lowerBound;  
  62.     }  
  63.   
  64.     if ((maxNumOfPixels == -1) && (minSideLength == -1)) {  
  65.         return 1;  
  66.     } else if (minSideLength == -1) {  
  67.         return lowerBound;  
  68.     } else {  
  69.         return upperBound;  
  70.     }  
  71. }  
  72.    


[java]  view plain  copy
  1. public class BitmapUtil {  
  2.   
  3.     public static int calculateInSampleSize(BitmapFactory.Options options,  int reqWidth, int reqHeight) {      
  4.         // 源图片的高度和宽度         
  5.         final int height = options.outHeight;        
  6.         final int width = options.outWidth;         
  7.         int inSampleSize = 1;       
  8.         if (height > reqHeight || width > reqWidth) {          
  9.             // 计算出实际宽高和目标宽高的比率      
  10.             final int heightRatio = Math.round((float) height / (float) reqHeight);        
  11.             final int widthRatio = Math.round((float) width / (float) reqWidth);         
  12.             // 选择宽和高中最小的比率作为inSampleSize的值,这样可以保证最终图片的宽和高           // 一定都会大于等于目标的宽和高。     
  13.             inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;      
  14.             }     
  15.         return inSampleSize;   
  16.           
  17.     }  
  18.     public Bitmap decodeSampleFromSD(String path,int sdwidth,int sdheight){  
  19.         BitmapFactory.Options options=new BitmapFactory.Options();  
  20.         options.inJustDecodeBounds=true;  
  21.           
  22.         BitmapFactory.decodeFile(path, options);  
  23.         options.inSampleSize=calculateInSampleSize(options, sdwidth, sdheight);  
  24.         options.inJustDecodeBounds=false;  
  25.         options.inDither=false;  
  26.         options.inPreferredConfig=Bitmap.Config.ARGB_8888;  
  27.           
  28.           
  29.         return BitmapFactory.decodeFile(path, options);  
  30.     }  
  31. }  




第一:我们先看下质量压缩方法:

 

  1. private Bitmap compressImage(Bitmap image) {  
  2.   
  3.         ByteArrayOutputStream baos = new ByteArrayOutputStream();  
  4.         image.compress(Bitmap.CompressFormat.JPEG, 100, baos);//质量压缩方法,这里100表示不压缩,把压缩后的数据存放到baos中  
  5.         int options = 100;  
  6.         while ( baos.toByteArray().length / 1024>100) {  //循环判断如果压缩后图片是否大于100kb,大于继续压缩         
  7.             baos.reset();//重置baos即清空baos  
  8.             image.compress(Bitmap.CompressFormat.JPEG, options, baos);//这里压缩options%,把压缩后的数据存放到baos中  
  9.             options -= 10;//每次都减少10  
  10.         }  
  11.         ByteArrayInputStream isBm = new ByteArrayInputStream(baos.toByteArray());//把压缩后的数据baos存放到ByteArrayInputStream中  
  12.         Bitmap bitmap = BitmapFactory.decodeStream(isBm, nullnull);//把ByteArrayInputStream数据生成图片  
  13.         return bitmap;  
  14.     }  
 

第二:图片按比例大小压缩方法(根据路径获取图片并压缩):

 

  1. private Bitmap getimage(String srcPath) {  
  2.         BitmapFactory.Options newOpts = new BitmapFactory.Options();  
  3.         //开始读入图片,此时把options.inJustDecodeBounds 设回true了  
  4.         newOpts.inJustDecodeBounds = true;  
  5.         Bitmap bitmap = BitmapFactory.decodeFile(srcPath,newOpts);//此时返回bm为空  
  6.           
  7.         newOpts.inJustDecodeBounds = false;  
  8.         int w = newOpts.outWidth;  
  9.         int h = newOpts.outHeight;  
  10.         //现在主流手机比较多是800*480分辨率,所以高和宽我们设置为  
  11.         float hh = 800f;//这里设置高度为800f  
  12.         float ww = 480f;//这里设置宽度为480f  
  13.         //缩放比。由于是固定比例缩放,只用高或者宽其中一个数据进行计算即可  
  14.         int be = 1;//be=1表示不缩放  
  15.         if (w > h && w > ww) {//如果宽度大的话根据宽度固定大小缩放  
  16.             be = (int) (newOpts.outWidth / ww);  
  17.         } else if (w < h && h > hh) {//如果高度高的话根据宽度固定大小缩放  
  18.             be = (int) (newOpts.outHeight / hh);  
  19.         }  
  20.         if (be <= 0)  
  21.             be = 1;  
  22.         newOpts.inSampleSize = be;//设置缩放比例  
  23.         //重新读入图片,注意此时已经把options.inJustDecodeBounds 设回false了  
  24.         bitmap = BitmapFactory.decodeFile(srcPath, newOpts);  
  25.         return compressImage(bitmap);//压缩好比例大小后再进行质量压缩  
  26.     }  
 

第三:图片按比例大小压缩方法(根据Bitmap图片压缩):

 

  1. private Bitmap comp(Bitmap image) {  
  2.       
  3.     ByteArrayOutputStream baos = new ByteArrayOutputStream();         
  4.     image.compress(Bitmap.CompressFormat.JPEG, 100, baos);  
  5.     if( baos.toByteArray().length / 1024>1024) {//判断如果图片大于1M,进行压缩避免在生成图片(BitmapFactory.decodeStream)时溢出    
  6.         baos.reset();//重置baos即清空baos  
  7.         image.compress(Bitmap.CompressFormat.JPEG, 50, baos);//这里压缩50%,把压缩后的数据存放到baos中  
  8.     }  
  9.     ByteArrayInputStream isBm = new ByteArrayInputStream(baos.toByteArray());  
  10.     BitmapFactory.Options newOpts = new BitmapFactory.Options();  
  11.     //开始读入图片,此时把options.inJustDecodeBounds 设回true了  
  12.     newOpts.inJustDecodeBounds = true;  
  13.     Bitmap bitmap = BitmapFactory.decodeStream(isBm, null, newOpts);  
  14.     newOpts.inJustDecodeBounds = false;  
  15.     int w = newOpts.outWidth;  
  16.     int h = newOpts.outHeight;  
  17.     //现在主流手机比较多是800*480分辨率,所以高和宽我们设置为  
  18.     float hh = 800f;//这里设置高度为800f  
  19.     float ww = 480f;//这里设置宽度为480f  
  20.     //缩放比。由于是固定比例缩放,只用高或者宽其中一个数据进行计算即可  
  21.     int be = 1;//be=1表示不缩放  
  22.     if (w > h && w > ww) {//如果宽度大的话根据宽度固定大小缩放  
  23.         be = (int) (newOpts.outWidth / ww);  
  24.     } else if (w < h && h > hh) {//如果高度高的话根据宽度固定大小缩放  
  25.         be = (int) (newOpts.outHeight / hh);  
  26.     }  
  27.     if (be <= 0)  
  28.         be = 1;  
  29.     newOpts.inSampleSize = be;//设置缩放比例  
  30.     //重新读入图片,注意此时已经把options.inJustDecodeBounds 设回false了  
  31.     isBm = new ByteArrayInputStream(baos.toByteArray());  
  32.     bitmap = BitmapFactory.decodeStream(isBm, null, newOpts);  
  33.     return compressImage(bitmap);//压缩好比例大小后再进行质量压缩  
  34. }  

你可能感兴趣的:(android 图片压缩)