Android-图片的压缩(质量压缩和尺寸压缩)

复习一下图片的压缩知识,今天来做一个总结。 参考:https://blog.csdn.net/baidu_38477614/article/details/78901107

理论概括

1.图片存在的几种形式:

  • File,存在于我们的磁盘中,我们通常说的图片大小。
  • Stream即流的形式,比如我们上传网络图片。
  • Bitmap,就是我们通常指内存中图片的大小。

2. 什么是质量压缩?

图片的质量压缩,会改变图片在磁盘中的大小(File文件的大小),不能改变图片在加载时,在内存中的大小。 原理是:通过算法扣掉(同化)了 图片中的一些某个点附近相近的像素,达到降低质量 减少 文件大小的目的。 应用场景:图片的上传。

3.什么是尺寸压缩?

图片的尺寸压缩是指:按照一定的倍数对图片减少单位尺寸的像素值,可以改变图片在内存中的大小,不改变图片在磁盘中的大小。 原理是:通过减少单位尺寸的像素值,真正意义上的降低像素值。 应用场景:用户头像的缩略图。

实战

我们的界面也很简单,就是两个按钮,分别是拍照和相册选择,一个ImageView,用来显示压缩后的图片,如图:

image.png

由于我们这里只讲图片的压缩,关于再次之前如何获取图片返回的URI和高低版本适配7.0等问题,我们这里不说,我之前写过文章,Android-图片的选择,裁剪,压缩,适配高版本,这里就不说了。 我们直接从图片获取到拍照或者相册返回的URI开始说起,上图:

image.png

质量压缩

那我们就先看bitmapCompress()这个质量压缩的方法。

/**
     * 这里我们生成了一个Pic文件夹,在下面放了我们质量压缩后的图片,用于和原图对比
     * 压缩图片使用Bitmap.compress(),这里是质量压缩
     */
    public void bitmapCompress(Uri uriClipUri) {
        try {
            //裁剪后的图像转成BitMap
            //photoBitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(uriClipUri));
            photoBitmap = MediaStore.Images.Media.getBitmap(getContentResolver(), uriClipUri);
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        //创建路径
        String path = Environment.getExternalStorageDirectory()
                .getPath() + "/Pic";
        //获取外部储存目录
        file = new File(path);
        //创建新目录, 创建此抽象路径名指定的目录,包括创建必需但不存在的父目录。
        file.mkdirs();
        //以当前时间重新命名文件
        long i = System.currentTimeMillis();
        //生成新的文件
        file = new File(file.toString() + "/" + i + ".png");
        Log.e("fileNew", file.getPath());
        //创建输出流
        OutputStream out = null;
        try {
            out = new FileOutputStream(file.getPath());
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
        //压缩文件,返回结果,参数分别是压缩的格式,压缩质量的百分比,输出流
        boolean bCompress = photoBitmap.compress(Bitmap.CompressFormat.JPEG, 50, out);

        try {
            photoBitmap = MediaStore.Images.Media.getBitmap(getContentResolver(),Uri.fromFile(file));
        } catch (IOException e) {
            e.printStackTrace();
        }
        imageView.setImageBitmap(photoBitmap);
    }

我们看一眼结果: 结果是Imageview没有正常加载图片,怎么回事?难道图片没有生成,文件创建失败? 我们看一眼原图片和压缩目录(Pic)下有没有文件: 原文件:

压缩后的文件:

可以看到原文件和压缩后的文件都生成了,而且也从6.61M压缩为了1.52M,那为什么图片不正常显示呢?,在看一眼日志:

image.png

大家明白了吧,这个结果也和我们之前说的质量压缩只是改变磁盘中的文件大小,并不能改变加载时内存中的图片大小

尺寸压缩

尺寸压缩的方法:

 Bitmap photoBitmap;
    File file;
    /**
     * 压缩图片使用,采用BitmapFactory.decodeFile。这里是尺寸压缩
     */

    public void bitmapFactory(Uri imageUri){
        String[] filePathColumns = {MediaStore.Images.Media.DATA};
        Cursor c = getContentResolver().query(imageUri, filePathColumns, null, null, null);
        c.moveToFirst();
        int columnIndex = c.getColumnIndex(filePathColumns[0]);
        String imagePath = c.getString(columnIndex);
        c.close();

        // 配置压缩的参数
        BitmapFactory.Options options = new BitmapFactory.Options();
        options.inJustDecodeBounds = true; //获取当前图片的边界大小,而不是将整张图片载入在内存中,避免内存溢出
        BitmapFactory.decodeFile(imagePath, options);
        options.inJustDecodeBounds = false;
        ////inSampleSize的作用就是可以把图片的长短缩小inSampleSize倍,所占内存缩小inSampleSize的平方
        options.inSampleSize = caculateSampleSize(options,500,50);
        Bitmap bm = BitmapFactory.decodeFile(imagePath, options); // 解码文件
        imageView.setImageBitmap(bm);
    }

    /**
     * 计算出所需要压缩的大小
     * @param options
     * @param reqWidth  我们期望的图片的宽,单位px
     * @param reqHeight 我们期望的图片的高,单位px
     * @return
     */
    private int caculateSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
        int sampleSize = 1;
        int picWidth = options.outWidth;
        int picHeight = options.outHeight;
        if (picWidth > reqWidth || picHeight > reqHeight) {
            int halfPicWidth = picWidth / 2;
            int halfPicHeight = picHeight / 2;
            while (halfPicWidth / sampleSize > reqWidth || halfPicHeight / sampleSize > reqHeight) {
                sampleSize *= 2;
            }
        }
        return sampleSize;
    }

结果:

图片正常显示,磁盘中图片的大小并没有改变,只是改变了加载时内存中的图片大小。

补充

  • 质量压缩无法避免oom,但可以改变图片在磁盘中或者说是File文件的大小,尺寸压缩可以避免OOM,但不改变图片本身的大小,只改变加载是在内存中的大小,即bitmap.
  • 质量压缩我们的主要方法是:MediaStore.Images.Media.getBitmap或者BitmapFactory.decodeStream;尺寸压缩我们用到的方法是:BitmapFactory.decodeFile

主要就说完了,我们在实际运用中可以把这两个方法作为工具类,随时调用。 demo上传github,地址:图片的质量和尺寸压缩

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏小灰灰

zxing二维码生成服务之深度定制

二维码生成服务之深度定制 之前写了一篇二维码服务定制的博文,现在则在之前的基础上,再进一步,花样的实现深度定制的需求,我们的目标是二维码上的一切都是可以由用户...

56260
来自专栏Google Dart

Flutter中构建布局 顶

然后本指南回过头来解释Flutter的布局方法,并说明如何在屏幕上放置一个小部件。 在讨论如何水平和垂直放置小部件之后,会介绍一些最常见的布局小部件。

68710
来自专栏于晓飞的专栏

ImageLoader 相关知识点

我们不能够通过构造函数创建Bitmap对象。如果需要将图片转成Bitmap对象加载到内存中,就需要使用BitmapFactory类。BitmapFactory跟...

11110
来自专栏技术总结

YYImage框架瞧一瞧

建议查看原文:https://www.jianshu.com/p/83edaeeb5851(不定时更新)

16530
来自专栏GIS讲堂

jquery判断一个div的边界是否超出另外一个div的边界

摘要:本文简单介绍jquery判断一个div的边界是否超出另外一个div的边界,如果超出边界做出相应的处理。

11740
来自专栏郭霖

Android高效加载大图、多图解决方案,有效避免程序OOM

高效加载大图片 我们在编写Android程序的时候经常要用到许多图片,不同图片总是会有不同的形状、不同的大小,但在大多数情况下,这些图片都会大于我们程序所需要的...

57170
来自专栏阿炬.NET

FineUIMvc表格数据库分页,使用CYQ.Data组件

42780
来自专栏DeveWork

WordPress 3.9+的 TinyMCE 4 编辑器增强开发

从WordPress 3.9版本后,WordPress 默认的编辑器 TinyMCE 随之升级到了版本4,带来的问题以前在默认编辑器上的增强开发的效果可能失效。...

24860
来自专栏阿炬.NET

FineUIMvc表格数据库分页,使用CYQ.Data组件

15140
来自专栏wOw的Android小站

[Android] 使用MediaProjection截屏

Android5.0以上提供了MediaProjection,方便截屏录屏等功能。

2.5K10

扫码关注云+社区

领取腾讯云代金券