在网上调查了图片压缩的方法并实装后,大致上可以认为有两类压缩:质量压缩(不改变图片的尺寸)和尺寸压缩(相当于是像素上的压缩);质量压缩一般可用于上传大图前的处理,这样就可以节省一定的流量,毕竟现在的手机拍照都能达到3M左右了,尺寸压缩一般可用于生成缩略图。
两种方法都实装在了我的项目中,结果却发现在质量压缩的模块中,本来1.9M的图片压缩后反而变成3M多了,很是奇怪,再做了进一步调查终于知道原因了。下面这个博客说的比较清晰:

android图片压缩总结

总结来看,图片有三种存在形式:硬盘上时是file,网络传输时是stream,内存中是stream或bitmap,所谓的质量压缩,它其实只能实现对file的影响,你可以把一个file转成bitmap再转成file,或者直接将一个bitmap转成file时,这个最终的file是被压缩过的,但是中间的bitmap并没有被压缩(或者说几乎没有被压缩,我不确定),因为bigmap在内存中的大小是按像素计算的,也就是width * height,对于质量压缩,并不会改变图片的像素,所以就算质量被压缩了,但是bitmap在内存的占有率还是没变小,但你做成file时,它确实变小了;

而尺寸压缩由于是减小了图片的像素,所以它直接对bitmap产生了影响,当然最终的file也是相对的变小了;

最后把自己总结的工具类贴出来:

[java] view plaincopy
  1. import java.io.ByteArrayInputStream;
  2. import java.io.ByteArrayOutputStream;
  3. import java.io.File;
  4. import java.io.FileNotFoundException;
  5. import java.io.FileOutputStream;
  6. import java.io.IOException;
  7. import android.graphics.Bitmap;
  8. import android.graphics.Bitmap.Config;
  9. import android.graphics.BitmapFactory;
  10. /**
  11. * Image compress factory class
  12. *
  13. * @author
  14. *
  15. */
  16. public class ImageFactory {
  17. /**
  18. * Get bitmap from specified image path
  19. *
  20. * @param imgPath
  21. * @return
  22. */
  23. public Bitmap getBitmap(String imgPath) {
  24. // Get bitmap through image path
  25. BitmapFactory.Options newOpts = new BitmapFactory.Options();
  26. newOpts.inJustDecodeBounds = false;
  27. newOpts.inPurgeable = true;
  28. newOpts.inInputShareable = true;
  29. // Do not compress
  30. newOpts.inSampleSize = 1;
  31. newOpts.inPreferredConfig = Config.RGB_565;
  32. return BitmapFactory.decodeFile(imgPath, newOpts);
  33. }
  34. /**
  35. * Store bitmap into specified image path
  36. *
  37. * @param bitmap
  38. * @param outPath
  39. * @throws FileNotFoundException
  40. */
  41. public void storeImage(Bitmap bitmap, String outPath) throws FileNotFoundException {
  42. FileOutputStream os = new FileOutputStream(outPath);
  43. bitmap.compress(Bitmap.CompressFormat.JPEG, 100, os);
  44. }
  45. /**
  46. * Compress image by pixel, this will modify image width/height.
  47. * Used to get thumbnail
  48. *
  49. * @param imgPath image path
  50. * @param pixelW target pixel of width
  51. * @param pixelH target pixel of height
  52. * @return
  53. */
  54. public Bitmap ratio(String imgPath, float pixelW, float pixelH) {
  55. BitmapFactory.Options newOpts = new BitmapFactory.Options();
  56. // 开始读入图片,此时把options.inJustDecodeBounds 设回true,即只读边不读内容
  57. newOpts.inJustDecodeBounds = true;
  58. newOpts.inPreferredConfig = Config.RGB_565;
  59. // Get bitmap info, but notice that bitmap is null now
  60. Bitmap bitmap = BitmapFactory.decodeFile(imgPath,newOpts);
  61. newOpts.inJustDecodeBounds = false;
  62. int w = newOpts.outWidth;
  63. int h = newOpts.outHeight;
  64. // 想要缩放的目标尺寸
  65. float hh = pixelH;// 设置高度为240f时,可以明显看到图片缩小了
  66. float ww = pixelW;// 设置宽度为120f,可以明显看到图片缩小了
  67. // 缩放比。由于是固定比例缩放,只用高或者宽其中一个数据进行计算即可
  68. int be = 1;//be=1表示不缩放
  69. if (w > h && w > ww) {//如果宽度大的话根据宽度固定大小缩放
  70. be = (int) (newOpts.outWidth / ww);
  71. } else if (w < h && h > hh) {//如果高度高的话根据宽度固定大小缩放
  72. be = (int) (newOpts.outHeight / hh);
  73. }
  74. if (be <= 0) be = 1;
  75. newOpts.inSampleSize = be;//设置缩放比例
  76. // 开始压缩图片,注意此时已经把options.inJustDecodeBounds 设回false了
  77. bitmap = BitmapFactory.decodeFile(imgPath, newOpts);
  78. // 压缩好比例大小后再进行质量压缩
  79. //        return compress(bitmap, maxSize); // 这里再进行质量压缩的意义不大,反而耗资源,删除
  80. return bitmap;
  81. }
  82. /**
  83. * Compress image by size, this will modify image width/height.
  84. * Used to get thumbnail
  85. *
  86. * @param image
  87. * @param pixelW target pixel of width
  88. * @param pixelH target pixel of height
  89. * @return
  90. */
  91. public Bitmap ratio(Bitmap image, float pixelW, float pixelH) {
  92. ByteArrayOutputStream os = new ByteArrayOutputStream();
  93. image.compress(Bitmap.CompressFormat.JPEG, 100, os);
  94. if( os.toByteArray().length / 1024>1024) {//判断如果图片大于1M,进行压缩避免在生成图片(BitmapFactory.decodeStream)时溢出
  95. os.reset();//重置baos即清空baos
  96. image.compress(Bitmap.CompressFormat.JPEG, 50, os);//这里压缩50%,把压缩后的数据存放到baos中
  97. }
  98. ByteArrayInputStream is = new ByteArrayInputStream(os.toByteArray());
  99. BitmapFactory.Options newOpts = new BitmapFactory.Options();
  100. //开始读入图片,此时把options.inJustDecodeBounds 设回true了
  101. newOpts.inJustDecodeBounds = true;
  102. newOpts.inPreferredConfig = Config.RGB_565;
  103. Bitmap bitmap = BitmapFactory.decodeStream(is, null, newOpts);
  104. newOpts.inJustDecodeBounds = false;
  105. int w = newOpts.outWidth;
  106. int h = newOpts.outHeight;
  107. float hh = pixelH;// 设置高度为240f时,可以明显看到图片缩小了
  108. float ww = pixelW;// 设置宽度为120f,可以明显看到图片缩小了
  109. //缩放比。由于是固定比例缩放,只用高或者宽其中一个数据进行计算即可
  110. int be = 1;//be=1表示不缩放
  111. if (w > h && w > ww) {//如果宽度大的话根据宽度固定大小缩放
  112. be = (int) (newOpts.outWidth / ww);
  113. } else if (w < h && h > hh) {//如果高度高的话根据宽度固定大小缩放
  114. be = (int) (newOpts.outHeight / hh);
  115. }
  116. if (be <= 0) be = 1;
  117. newOpts.inSampleSize = be;//设置缩放比例
  118. //重新读入图片,注意此时已经把options.inJustDecodeBounds 设回false了
  119. is = new ByteArrayInputStream(os.toByteArray());
  120. bitmap = BitmapFactory.decodeStream(is, null, newOpts);
  121. //压缩好比例大小后再进行质量压缩
  122. //      return compress(bitmap, maxSize); // 这里再进行质量压缩的意义不大,反而耗资源,删除
  123. return bitmap;
  124. }
  125. /**
  126. * Compress by quality,  and generate image to the path specified
  127. *
  128. * @param image
  129. * @param outPath
  130. * @param maxSize target will be compressed to be smaller than this size.(kb)
  131. * @throws IOException
  132. */
  133. public void compressAndGenImage(Bitmap image, String outPath, int maxSize) throws IOException {
  134. ByteArrayOutputStream os = new ByteArrayOutputStream();
  135. // scale
  136. int options = 100;
  137. // Store the bitmap into output stream(no compress)
  138. image.compress(Bitmap.CompressFormat.JPEG, options, os);
  139. // Compress by loop
  140. while ( os.toByteArray().length / 1024 > maxSize) {
  141. // Clean up os
  142. os.reset();
  143. // interval 10
  144. options -= 10;
  145. image.compress(Bitmap.CompressFormat.JPEG, options, os);
  146. }
  147. // Generate compressed image file
  148. FileOutputStream fos = new FileOutputStream(outPath);
  149. fos.write(os.toByteArray());
  150. fos.flush();
  151. fos.close();
  152. }
  153. /**
  154. * Compress by quality,  and generate image to the path specified
  155. *
  156. * @param imgPath
  157. * @param outPath
  158. * @param maxSize target will be compressed to be smaller than this size.(kb)
  159. * @param needsDelete Whether delete original file after compress
  160. * @throws IOException
  161. */
  162. public void compressAndGenImage(String imgPath, String outPath, int maxSize, boolean needsDelete) throws IOException {
  163. compressAndGenImage(getBitmap(imgPath), outPath, maxSize);
  164. // Delete original file
  165. if (needsDelete) {
  166. File file = new File (imgPath);
  167. if (file.exists()) {
  168. file.delete();
  169. }
  170. }
  171. }
  172. /**
  173. * Ratio and generate thumb to the path specified
  174. *
  175. * @param image
  176. * @param outPath
  177. * @param pixelW target pixel of width
  178. * @param pixelH target pixel of height
  179. * @throws FileNotFoundException
  180. */
  181. public void ratioAndGenThumb(Bitmap image, String outPath, float pixelW, float pixelH) throws FileNotFoundException {
  182. Bitmap bitmap = ratio(image, pixelW, pixelH);
  183. storeImage( bitmap, outPath);
  184. }
  185. /**
  186. * Ratio and generate thumb to the path specified
  187. *
  188. * @param image
  189. * @param outPath
  190. * @param pixelW target pixel of width
  191. * @param pixelH target pixel of height
  192. * @param needsDelete Whether delete original file after compress
  193. * @throws FileNotFoundException
  194. */
  195. public void ratioAndGenThumb(String imgPath, String outPath, float pixelW, float pixelH, boolean needsDelete) throws FileNotFoundException {
  196. Bitmap bitmap = ratio(imgPath, pixelW, pixelH);
  197. storeImage( bitmap, outPath);
  198. // Delete original file
  199. if (needsDelete) {
  200. File file = new File (imgPath);
  201. if (file.exists()) {
  202. file.delete();
  203. }
  204. }
  205. }
  206. }
延伸阅读

android图片压缩总结

一.图片的存在形式

1.文件形式(即以二进制形式存在于硬盘上)
2.流的形式(即以二进制形式存在于内存中)
3.Bitmap形式
这三种形式的区别: 文件形式和流的形式对图片体积大小并没有影响,也就是说,如果你手机SD卡上的如果是100K,那么通过流的形式读到内存中,也一定是占100K的内存,注意是流的形式,不是Bitmap的形式,当图片以Bitmap的形式存在时,其占用的内存会瞬间变大, 我试过500K文件形式的图片加载到内存,以Bitmap形式存在时,占用内存将近10M,当然这个增大的倍数并不是固定的
检测图片三种形式大小的方法:
文件形式: file.length()
流的形式: 讲图片文件读到内存输入流中,看它的byte数
Bitmap:    bitmap.getByteCount()

二.常见的压缩方式

[java] view plaincopy
  1. public static void compressBmpToFile(Bitmap bmp,File file){
  2. ByteArrayOutputStream baos = new ByteArrayOutputStream();
  3. int options = 80;//个人喜欢从80开始,
  4. bmp.compress(Bitmap.CompressFormat.JPEG, options, baos);
  5. while (baos.toByteArray().length / 1024 > 100) {
  6. baos.reset();
  7. options -= 10;
  8. bmp.compress(Bitmap.CompressFormat.JPEG, options, baos);
  9. }
  10. try {
  11. FileOutputStream fos = new FileOutputStream(file);
  12. fos.write(baos.toByteArray());
  13. fos.flush();
  14. fos.close();
  15. } catch (Exception e) {
  16. e.printStackTrace();
  17. }
  18. }
方法说明: 该方法是压缩图片的质量, 注意它不会减少图片的像素,比方说, 你的图片是300K的, 1280*700像素的, 经过该方法压缩后, File形式的图片是在100以下, 以方便上传服务器, 但是你BitmapFactory.decodeFile到内存中,变成Bitmap时,它的像素仍然是1280*700, 计算图片像素的方法是 bitmap.getWidth()和bitmap.getHeight(), 图片是由像素组成的, 每个像素又包含什么呢? 熟悉PS的人知道, 图片是有色相,明度和饱和度构成的. 
该方法的官方文档也解释说, 它会让图片重新构造, 但是有可能图像的位深(即色深)和每个像素的透明度会变化,JPEG onlysupports opaque(不透明), 也就是说以jpeg格式压缩后, 原来图片中透明的元素将消失.所以这种格式很可能造成失真

既然它是改变了图片的显示质量, 达到了对File形式的图片进行压缩, 图片的像素没有改变的话, 那重新读取经过压缩的file为Bitmap时, 它占用的内存并不会少.(不相信的可以试试)

因为: bitmap.getByteCount() 是计算它的像素所占用的内存, 请看官方解释: Returns the number of bytes used to store this bitmap's pixels.

2.   将图片从本地读到内存时,进行压缩 ,即图片从File形式变为Bitmap形式

       特点: 通过设置采样率, 减少图片的像素, 达到对内存中的Bitmap进行压缩
       先看一个方法: 该方法是对内存中的Bitmap进行质量上的压缩, 由上面的理论可以得出该方法是无效的, 而且也是没有必要的,因为你已经将它读到内存中了,再压缩多此一举, 尽管在获取系统相册图片时,某些手机会直接返回一个Bitmap,但是这种情况下, 返回的Bitmap都是经过压缩的, 它不可能直接返回一个原声的Bitmap形式的图片, 后果可想而知
[java] view plaincopy
  1. private Bitmap compressBmpFromBmp(Bitmap image) {
  2. ByteArrayOutputStream baos = new ByteArrayOutputStream();
  3. int options = 100;
  4. image.compress(Bitmap.CompressFormat.JPEG, 100, baos);
  5. while (baos.toByteArray().length / 1024 > 100) {
  6. baos.reset();
  7. options -= 10;
  8. image.compress(Bitmap.CompressFormat.JPEG, options, baos);
  9. }
  10. ByteArrayInputStream isBm = new ByteArrayInputStream(baos.toByteArray());
  11. Bitmap bitmap = BitmapFactory.decodeStream(isBm, null, null);
  12. return bitmap;
  13. }

再看一个方法:

[java] view plaincopy
  1. private Bitmap compressImageFromFile(String srcPath) {
  2. BitmapFactory.Options newOpts = new BitmapFactory.Options();
  3. newOpts.inJustDecodeBounds = true;//只读边,不读内容
  4. Bitmap bitmap = BitmapFactory.decodeFile(srcPath, newOpts);
  5. newOpts.inJustDecodeBounds = false;
  6. int w = newOpts.outWidth;
  7. int h = newOpts.outHeight;
  8. float hh = 800f;//
  9. float ww = 480f;//
  10. int be = 1;
  11. if (w > h && w > ww) {
  12. be = (int) (newOpts.outWidth / ww);
  13. } else if (w < h && h > hh) {
  14. be = (int) (newOpts.outHeight / hh);
  15. }
  16. if (be <= 0)
  17. be = 1;
  18. newOpts.inSampleSize = be;//设置采样率
  19. newOpts.inPreferredConfig = Config.ARGB_8888;//该模式是默认的,可不设
  20. newOpts.inPurgeable = true;// 同时设置才会有效
  21. newOpts.inInputShareable = true;//。当系统内存不够时候图片自动被回收
  22. bitmap = BitmapFactory.decodeFile(srcPath, newOpts);
  23. //      return compressBmpFromBmp(bitmap);//原来的方法调用了这个方法企图进行二次压缩
  24. //其实是无效的,大家尽管尝试
  25. return bitmap;
  26. }
方法说明: 该方法就是对Bitmap形式的图片进行压缩, 也就是通过设置采样率, 减少Bitmap的像素, 从而减少了它所占用的内存

android 图片压缩总结1相关推荐

  1. Android图片压缩(质量压缩和尺寸压缩)

    在网上调查了图片压缩的方法并实装后,大致上可以认为有两类压缩:质量压缩(不改变图片的尺寸)和尺寸压缩(相当于是像素上的压缩):质量压缩一般可用于上传大图前的处理,这样就可以节省一定的流量,毕竟现在的手 ...

  2. Android 图片压缩、照片选择、裁剪,上传、一整套图片解决方案

    1.Android一整套图片解决方案 http://mp.weixin.qq.com/s?__biz=MzAxMTI4MTkwNQ==&mid=2650820998&idx=1& ...

  3. Android 图片压缩器

    概述 Android 图片压缩器:一款高效的图片压缩器库,支持批量压缩,异步压缩.多线程多任务压缩,压缩比设置等特性. 详细 代码下载:http://www.demodashi.com/demo/12 ...

  4. Android图片压缩尺寸和质量

    Android在处理图片时,如果不进行压缩处理,很容易就出现OOM内存溢出(OutOfMemory)问题,所以无论是第三方图片加载还是自己在处理图片时,都要进行压缩处理. Android系统中,一张图 ...

  5. Android图片压缩库——libjpeg-turbo

    为何Android图片压缩效率比IOS低质量差 为什么Android的图片压缩质量要比iPhone的压缩质量差很多,这是因为Android底层犯的一个小错误:libjpeg.并且这个错误一直持续到了今 ...

  6. 最详细的Android图片压缩攻略

    Mr.Louis的博客地址: https://blog.csdn.net/weixin_44005563 最近在研究图片压缩原理,看了大量资料,从上层尺寸压缩.质量压缩原理到下层的哈夫曼压缩,走成华大 ...

  7. android 图片压缩,bitmap压缩总结

    1.    图片压缩相关概念 在android开发中,图片加载到内存中通常需要占用大量的内存,导致app性能过度消耗,严重的话可能会导致OOM异常,所以对图片进行优化就尤为重要. android图片压 ...

  8. Android 图片压缩各种方式

    前言:由于公司项目当中需要用到压缩这块的相应技术,之前也做过的图片压缩都不是特别的理想, 所以这次花了很多心思,仔细研究和在网上找到了很多相对应的资料.为了就是 以后再做的时候直接拿来用就可以了! 第 ...

  9. 可能是最详细的Android图片压缩原理分析(二)—— 鲁班压缩算法解析

    本篇文章已授权微信公众号guolin_blog(郭霖)独家发布 稀土掘金链接 前言 通过上一篇,我们了解了一些关于图片压缩的基础知识,这篇文章我们主要讲解一下鲁班压缩的算法逻辑,很多博客都是从Gith ...

最新文章

  1. centos7中nfs文件系统的使用
  2. hackerrank杂记
  3. java类载入器——ClassLoader
  4. 【IM】关于集成学习Bagging和Boosting的理解
  5. ds18b20温度转换指令_DS18B20温度传感器(附代码并浅谈与或运算)
  6. XMNetworking 网络库的设计与使用
  7. 逆序数问题,用归并排序而非树状数组求解
  8. React和Vue的Chrome扩展工具安装
  9. php web教程视频教程下载,Web全栈 PHP+React系列视频教程下载
  10. html5元件的作用,寄存器的作用是什么
  11. JavaScript技巧写法
  12. chrome pdf viewer 参数
  13. DVI 口转 VGA 手工接线方法
  14. python百例练习之第三例-字典dict()
  15. 电驴怎么显示服务器列表,(转)如何更新电驴服务器列表(eMule Server List)
  16. html的代码怎麼格式化,怎么格式化html代码? Dreamweaver格式化html代码的技巧
  17. matlab二阶阻尼震荡衰减,二阶欠阻尼电路的零响应输入-MATLAB课程设计.doc
  18. 登录超时连不上信息服务器,我的侠客进不去怎么办 服务器登录连接超时请稍后再试解决方法...
  19. SpringBoot精藏(五)SpringBoot整合mybatis
  20. 浙江大学竺可桢计算机专业,2018浙江大学竺可桢学院进行新生选拔面试

热门文章

  1. org.eclipse.jdt版本更新导致包引入问题
  2. jQuery 模态框
  3. 关于物联网应用的一些预测
  4. FX5U modbustcp通讯
  5. Python 入门学习路线
  6. 学it中的python怎么样_长安大专网教学历
  7. 区块链会计案例_区块链会计行业 区块链会计应用案例
  8. iOS 防止截屏、录屏技术
  9. ios申请企业开发者账号的代理_iOS企业级开发者账号申请
  10. java写netcdf_[转]netcdf入门