热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Android图片压缩相机相册获取照片高质量压缩不会失真模糊含demo

先说一下 为什么进行图片压缩 主要有两个方面的原因 一是为了优化交互  减少上传图片的等待时间 和响应时间 而是为了减少服务器和数据库压力 减少图片的储存占用空间 当然使用第三方存储的可以忽略 例

先说一下 为什么进行图片压缩 主要有两个方面的原因 一是为了优化交互  减少上传图片的等待时间 和响应时间 而是为了减少服务器和数据库压力 减少图片的储存占用空间 当然使用第三方存储的可以忽略 例如七牛 当然我们也是能省则省嘛

本人今天整理一下android 关于照片压缩问题 关于图片压缩好很多方式 例如说质量压缩 比例压缩 等等 

今天笔者阐述的是比例压缩与质量压缩相结合的方式 压缩效果是很明显的 并且压缩后的图片不会失真模糊 

关于权限问题以及7.0 file路径问题 demo 已经做好适配 这里就不过多的阐述了 下载demo 解压直接查看就好了 

下面就直接上代码 

xml 文件

android:layout_width=”match_parent”

android:layout_height=”match_parent”

android:gravity=”center”

android:orientation=”vertical”>

android:id=”@+id

_camere”

android:layout_width=”200dp”

android:layout_height=”50dp”

android:text=”CAMARE” />

android:id=”@+id

_gallery”

android:layout_width=”200dp”

android:layout_height=”50dp”

android:layout_marginTop=”10dp”

android:text=”GALLERY” />

android:id=”@+id/im_src”

android:layout_width=”300dp”

android:layout_height=”300dp”

android:layout_marginTop=”20dp”

android:rotation=”-10″

android:src=”@mipmap/ic_launcher” />

// 相机

    private void takeGallery() {

        out = new File(FileUtils.getFileName());        if (Build.VERSION.SDK_INT >= 24){

            outputFileUri = FileProvider.getUriForFile(MainActivity.this,this.getPackageName()+”.provider”, out);        }else {

            outputFileUri = Uri.fromFile(out);        }

        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);        intent.putExtra(MediaStore.EXTRA_OUTPUT, outputFileUri);        if (Build.VERSION.SDK_INT >= 24) {

            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);        }

        if (intent.resolveActivity(getPackageManager()) != null) {

            startActivityForResult(intent, TYPE_CAMERE);        }

    }

    // 相册

    private void takeCamere() {

        Intent intent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI);        intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,”image/*”);        startActivityForResult(intent, TYPE_GALLERY);    }

uri 转 StringFilePath

private String getRealPathFromURI(Uri contentURI) {

        String result = null;

        Cursor cursor = getContentResolver().query(contentURI, null, null, null, null);

        if (cursor == null) {

            result = contentURI.getPath();

        } else {

            if (cursor.moveToFirst()){

                int idx = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA);

                if (idx > -1){

                    result = cursor.getString(idx);

                }

                cursor.close();

            }

        }

        return result;

    }

 相机相册回调

@Override protected void onActivityResult(int requestCode, int resultCode, Intent data) {

        super.onActivityResult(requestCode, resultCode, data);

        if (resultCode!=RESULT_OK)

            return;

        String fileName = FileUtils.getFileName();

        switch (requestCode){

            case TYPE_CAMERE:

                Bitmap smallBitmap = BitmapUtils.getSmallBitmap(out.getAbsolutePath(), 480, 480,new File(fileName));

                img_src.setImageBitmap(smallBitmap);

                break;

            case TYPE_GALLERY:

                Bitmap smallBitmap1 = BitmapUtils.getSmallBitmap(getRealPathFromURI(data.getData()), 480, 480,new File(fileName));

                img_src.setImageBitmap(smallBitmap1);

                break;

        }

    }

// file 工具 

public class FileUtils {

        // Create an image file name        public static String getFileName(){

            String fileName  =null;

            long currentTimeMillis = System.currentTimeMillis();

            String pathName = Environment.getExternalStorageDirectory() + “/DCIM/”;

            File file = new File(pathName);

            file.mkdirs();

            fileName = pathName + currentTimeMillis+”.jpg”;

            return fileName;

        }

    }

压缩工具

public classBitmapUtils{ 

 // 计算比例 

 public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {

        // Raw height and width of image        final int height = options.outHeight;

        final int width = options.outWidth;

        int inSampleSize = 1;

        if (height > reqHeight || width > reqWidth) {

            // Calculate ratios of height and width to requested height and            // width           

         final int heightRatio = Math.round((float) height

                    / (float) reqHeight);

            final int widthRatio = Math.round((float) width / (float) reqWidth);

            inSampleSize = heightRatio

        }

        return inSampleSize;

    }

    /**    * 根据路径获得到图片并压缩返回bitmap用于显示    *    * @paramfilePath    * @return*/   

         public static Bitmap getSmallBitmap(String filePath, int reqWidth, int reqHeight,File imgPath) {

        final BitmapFactory.Options optiOns= new BitmapFactory.Options();

        options.inJustDecodeBounds = true;  //只返回图片的大小信息                BitmapFactory.decodeFile(filePath, options);

        // Calculate inSampleSize       

         options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);

        // Decode bitmap with inSampleSize set        

        options.inJustDecodeBounds = false;

        Bitmap bitmap = BitmapFactory.decodeFile(filePath, options);

        Log.d(“tag”, “getSmallBitmap: 生成bitmap—-“+filePath+”//”+bitmap);

        compressBmpToFile(bitmap,imgPath);

        return bitmap;

    }

    /**    * 质量压缩   

 * @parambmp bitmap 对象    

 * @paramfile 路径文件   

 */    

        public static void compressBmpToFile(Bitmap bmp,File file){

        ByteArrayOutputStream baos = new ByteArrayOutputStream();

        int optiOns= 80;

        bmp.compress(Bitmap.CompressFormat.JPEG, options, baos);

        while (baos.toByteArray().length / 1024 > 100) {

            baos.reset();

            options -= 10;

            bmp.compress(Bitmap.CompressFormat.JPEG, options, baos);

        }

        try {

            FileOutputStream fos = new FileOutputStream(file);

            fos.write(baos.toByteArray());

            fos.flush();

            fos.close();

        } catch (Exception e) {

            e.printStackTrace();

        }

    }

}

本文是采用 比例压缩 和 质量压缩 结合的方法进行图片压缩  压缩后 图片质量很好 不会有失真的情况

看一下效果

Android 图片压缩 相机相册获取照片 高质量压缩 不会失真模糊 含demo
压缩比例

压缩后

Android 图片压缩 相机相册获取照片 高质量压缩 不会失真模糊 含demo
压缩后

好了 到这里就结束了 欢迎私信和留言  demo 下载


推荐阅读
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • 本文介绍了解决Netty拆包粘包问题的一种方法——使用特殊结束符。在通讯过程中,客户端和服务器协商定义一个特殊的分隔符号,只要没有发送分隔符号,就代表一条数据没有结束。文章还提供了服务端的示例代码。 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • VScode格式化文档换行或不换行的设置方法
    本文介绍了在VScode中设置格式化文档换行或不换行的方法,包括使用插件和修改settings.json文件的内容。详细步骤为:找到settings.json文件,将其中的代码替换为指定的代码。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • 本文介绍了如何将CIM_DateTime解析为.Net DateTime,并分享了解析过程中可能遇到的问题和解决方法。通过使用DateTime.ParseExact方法和适当的格式字符串,可以成功解析CIM_DateTime字符串。同时还提供了关于WMI和字符串格式的相关信息。 ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了一种划分和计数油田地块的方法。根据给定的条件,通过遍历和DFS算法,将符合条件的地块标记为不符合条件的地块,并进行计数。同时,还介绍了如何判断点是否在给定范围内的方法。 ... [详细]
  • CF:3D City Model(小思维)问题解析和代码实现
    本文通过解析CF:3D City Model问题,介绍了问题的背景和要求,并给出了相应的代码实现。该问题涉及到在一个矩形的网格上建造城市的情景,每个网格单元可以作为建筑的基础,建筑由多个立方体叠加而成。文章详细讲解了问题的解决思路,并给出了相应的代码实现供读者参考。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 本文讨论了Kotlin中扩展函数的一些惯用用法以及其合理性。作者认为在某些情况下,定义扩展函数没有意义,但官方的编码约定支持这种方式。文章还介绍了在类之外定义扩展函数的具体用法,并讨论了避免使用扩展函数的边缘情况。作者提出了对于扩展函数的合理性的质疑,并给出了自己的反驳。最后,文章强调了在编写Kotlin代码时可以自由地使用扩展函数的重要性。 ... [详细]
author-avatar
手机用户2602940163
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有