Android-图片压缩详解:原理、方法与实践

简介: 前言在Android应用开发中,处理图片是一个非常常见的需求。然而,大尺寸和高质量的图片可能会占用大量内存,导致应用程序性能下降,甚至引发OOM(Out of Memory)错误。因此,对图片进行合适的压缩是非常重要的。本文将详细介绍Android图片压缩的原理、方法和实践。

前言

在Android应用开发中,处理图片是一个非常常见的需求。然而,大尺寸和高质量的图片可能会占用大量内存,导致应用程序性能下降,甚至引发OOM(Out of Memory)错误。因此,对图片进行合适的压缩是非常重要的。本文将详细介绍Android图片压缩的原理、方法和实践。


1. 图片压缩的原理

图片压缩的本质是降低图片的尺寸(像素)和质量(色彩),以减少图片占用的内存空间。图片压缩可以分为两类:有损压缩和无损压缩。


有损压缩:在压缩过程中,一些图像信息将被丢失,导致图片质量降低。然而,有损压缩能够实现更高的压缩比,从而节省更多的内存空间。JPEG是一种典型的有损压缩算法。

无损压缩:在压缩过程中,图像信息得到保留,图片质量不会降低。然而,无损压缩的压缩比通常较低。PNG是一种典型的无损压缩算法。

2. Android图片压缩的方法

在Android中,有多种方法可以对图片进行压缩。以下是一些常见的图片压缩方法:

2.1 BitmapFactory.Options

BitmapFactory.Options是Android系统提供的一个类,用于设置图片解码时的参数。通过设置BitmapFactory.Options的属性,我们可以实现图片的尺寸和质量压缩。


2.1.1 尺寸压缩

尺寸压缩是通过减少图片的像素来降低图片占用的内存空间。尺寸压缩可以在图片解码时进行,从而避免OOM错误。


在BitmapFactory.Options中,有一个属性inSampleSize,表示解码时的缩放比例。例如,inSampleSize=2表示图片的宽度和高度将减小一半,图片占用的内存空间将减小四分之一。


以下是一个使用BitmapFactory.Options进行尺寸压缩的示例:

public static Bitmap decodeSampledBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight) {
    BitmapFactory.Options options = new BitmapFactory.Options();
    options.inJustDecodeBounds = true;
    BitmapFactory.decodeResource(res, resId, options);
    options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
    options.inJustDecodeBounds = false;
    return BitmapFactory.decodeResource(res, resId,options);
}
public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
    int inSampleSize = 1;
    int width = options.outWidth;
    int height = options.outHeight;
    if (width > reqWidth || height > reqHeight) {
        int halfWidth = width / 2;
        int halfHeight = height / 2;
        while ((halfWidth / inSampleSize) >= reqWidth && (halfHeight / inSampleSize) >= reqHeight) {
            inSampleSize *= 2;
        }
    }
    return inSampleSize;
}

在这个示例中,我们首先使用options.inJustDecodeBounds=true计算inSampleSize,然后使用options.inJustDecodeBounds=false进行实际的图片解码。这样可以避免解码两次图片,提高性能。


2.1.2 质量压缩

质量压缩是通过降低图片的色彩保真度来减少图片占用的内存空间。质量压缩主要针对有损压缩算法,如JPEG。


在Android中,我们可以使用Bitmap.compress()方法对图片进行质量压缩。以下是一个使用Bitmap.compress()进行质量压缩的示例:

public static Bitmap compressBitmap(Bitmap bitmap, int quality) {
    ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
    bitmap.compress(Bitmap.CompressFormat.JPEG, quality, outputStream);
    byte[] bytes = outputStream.toByteArray();
    return BitmapFactory.decodeByteArray(bytes, 0, bytes.length);
}

在这个示例中,我们将原始Bitmap压缩为JPEG格式,并指定压缩质量。压缩质量的范围是0-100,数值越小,压缩比越高,图片质量越低。需要注意的是,质量压缩不会减少Bitmap占用的内存空间,但可以减少图片文件的存储空间。


2.2 第三方库

除了使用Android系统提供的方法,我们还可以使用第三方库对图片进行压缩。以下是一些常见的Android图片压缩库:


Luban:一个轻量级的图片压缩库,支持尺寸和质量压缩。Luban使用一种智能算法,可以在保证图片质量的前提下实现高压缩比。

Glide:一个强大的图片加载和处理库,支持多种压缩策略。Glide可以根据目标ImageView的尺寸自动进行尺寸压缩,也可以使用Glide.with(context).load(url).thumbnail(scale)方法进行质量压缩。

TinyPNG:一个支持PNG和JPEG格式的图片压缩服务。TinyPNG提供了一个Android SDK,可以方便地在Android应用中进行图片压缩。

3. 实践:Android图片压缩

下面我们将通过一个实际示例来演示如何在Android应用中实现图片压缩。这个示例将展示如何使用BitmapFactory.Options进行尺寸压缩和质量压缩,以及如何使用Glide库进行图片压缩。


3.1 创建一个新的Android项目

首先,在Android Studio中创建一个新的Android项目。在这个示例中,我们将使用一个简单的布局,包括一个ImageView和一个Button。用户可以点击Button来选择图片,应用将对图片进行压缩并显示在ImageView上。


3.2 修改布局文件

打开activity_main.xml布局文件,添加一个ImageView和一个Button。代码如下:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity">
    <ImageView
        android:id="@+id/imageView"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:scaleType="fitCenter" />
    <Button
        android:id="@+id/button_select_image"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:text="Select Image" />
</LinearLayout>

3.3 添加图片选择功能

打开MainActivity.java,为Button添加点击事件监听器。当用户点击Button时,我们将启动一个图片选择器,让用户选择一张图片。代码如下:

private static final int REQUEST_CODE_PICK_IMAGE = 1;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    Button buttonSelectImage = findViewById(R.id.button_select_image);
    buttonSelectImage.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            Intent intent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
            startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE);
        }
    });
}
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if (requestCode == REQUEST_CODE_PICK_IMAGE && resultCode == RESULT_OK && data != null) {
        Uri imageUri = data.getData();
        if (imageUri != null) {
            // Load and compress the image
        }
    }
}

3.4 使用BitmapFactory.Options进行图片压缩

onActivityResult()方法中,我们将对用户选择的图片进行尺寸压缩和质量压缩。首先,我们使用BitmapFactory.Options进行尺寸压缩:

private Bitmap decodeSampledBitmapFromUri(Context context, Uri uri, int reqWidth, int reqHeight) {
    InputStream inputStream = null;
    try {
        inputStream = context.getContentResolver().openInputStream(uri);
        BitmapFactory.Options options = new BitmapFactory.Options();
        options.inJustDecodeBounds = true;
        BitmapFactory.decodeStream(inputStream, null, options);
        options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
        options.inJustDecodeBounds = false;
        inputStream.close();
        inputStream = context.getContentResolver().openInputStream(uri);
        return BitmapFactory.decodeStream(inputStream, null, options);
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        try {
            if (inputStream != null) {
                inputStream.close();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    return null;
}

private Bitmap compressBitmap(Bitmap bitmap, int quality) {
    ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
    bitmap.compress(Bitmap.CompressFormat.JPEG, quality, outputStream);
    byte[] bytes = outputStream.toByteArray();
    return BitmapFactory.decodeByteArray(bytes, 0, bytes.length);
}

然后,我们将压缩后的图片显示在ImageView上:

ImageView imageView = findViewById(R.id.imageView);
Bitmap compressedBitmap = decodeSampledBitmapFromUri(this, imageUri, imageView.getWidth(), imageView.getHeight());
compressedBitmap = compressBitmap(compressedBitmap, 80);
imageView.setImageBitmap(compressedBitmap);

3.5 使用Glide库进行图片压缩

首先,在build.gradle文件中添加Glide的依赖:

implementation 'com.github.bumptech.glide:glide:4.12.0'
annotationProcessor 'com.github.bumptech.glide:compiler:4.12.0'

然后,我们可以使用Glide的asBitmap()方法对图片进行尺寸压缩。在onActivityResult()方法中,将之前的代码替换为以下代码:

ImageView imageView = findViewById(R.id.imageView);
Glide.with(this)
     .asBitmap()
     .load(imageUri)
     .into(imageView);

Glide会根据ImageView的尺寸自动进行尺寸压缩。如果需要进行质量压缩,可以使用.thumbnail()方法。

4. 总结

在本文中,我们详细介绍了Android图片压缩的原理、方法和实践。我们学习了如何使用BitmapFactory.Options进行尺寸压缩和质量压缩,以及如何使用第三方库进行图片压缩。希望本文能帮助你更好地理解和应用Android图片压缩技术。

相关文章
|
1月前
|
缓存 搜索推荐 Android开发
安卓开发中的自定义控件实践
【10月更文挑战第4天】在安卓开发的海洋中,自定义控件是那片璀璨的星辰。它不仅让应用界面设计变得丰富多彩,还提升了用户体验。本文将带你探索自定义控件的核心概念、实现过程以及优化技巧,让你的应用在众多竞争者中脱颖而出。
|
26天前
|
缓存 Java Shell
Android 系统缓存扫描与清理方法分析
Android 系统缓存从原理探索到实现。
50 15
Android 系统缓存扫描与清理方法分析
|
25天前
|
缓存 Java 数据库
Android的ANR原理
【10月更文挑战第18天】了解 ANR 的原理对于开发高质量的 Android 应用至关重要。通过合理的设计和优化,可以有效避免 ANR 的发生,提升应用的性能和用户体验。
53 8
|
12天前
|
前端开发 Android开发 UED
安卓应用开发中的自定义控件实践
【10月更文挑战第35天】在移动应用开发中,自定义控件是提升用户体验、增强界面表现力的重要手段。本文将通过一个安卓自定义控件的创建过程,展示如何从零开始构建一个具有交互功能的自定义视图。我们将探索关键概念和步骤,包括继承View类、处理测量与布局、绘制以及事件处理。最终,我们将实现一个简单的圆形进度条,并分析其性能优化。
|
1月前
|
XML 前端开发 Android开发
Android View的绘制流程和原理详细解说
Android View的绘制流程和原理详细解说
39 3
|
1月前
|
前端开发 搜索推荐 Android开发
安卓开发中的自定义控件实践
【10月更文挑战第4天】在安卓开发的世界里,自定义控件如同画家的画笔,能够绘制出独一无二的界面。通过掌握自定义控件的绘制技巧,开发者可以突破系统提供的界面元素限制,创造出既符合品牌形象又提供卓越用户体验的应用。本文将引导你了解自定义控件的核心概念,并通过一个简单的例子展示如何实现一个基本的自定义控件,让你的安卓应用在视觉和交互上与众不同。
|
1月前
|
测试技术 数据库 Android开发
深入解析Android架构组件——Jetpack的使用与实践
本文旨在探讨谷歌推出的Android架构组件——Jetpack,在现代Android开发中的应用。Jetpack作为一系列库和工具的集合,旨在帮助开发者更轻松地编写出健壮、可维护且性能优异的应用。通过详细解析各个组件如Lifecycle、ViewModel、LiveData等,我们将了解其原理和使用场景,并结合实例展示如何在实际项目中应用这些组件,提升开发效率和应用质量。
44 6
|
2月前
|
安全 Android开发 数据安全/隐私保护
探索安卓与iOS的安全性差异:技术深度分析与实践建议
本文旨在深入探讨并比较Android和iOS两大移动操作系统在安全性方面的不同之处。通过详细的技术分析,揭示两者在架构设计、权限管理、应用生态及更新机制等方面的安全特性。同时,针对这些差异提出针对性的实践建议,旨在为开发者和用户提供增强移动设备安全性的参考。
139 3
|
2月前
|
ARouter 测试技术 API
Android经典面试题之组件化原理、优缺点、实现方法?
本文介绍了组件化在Android开发中的应用,详细阐述了其原理、优缺点及实现方式,包括模块化、接口编程、依赖注入、路由机制等内容,并提供了具体代码示例。
47 2
|
1月前
|
Java 调度 Android开发
Android面试题之Kotlin中async 和 await实现并发的原理和面试总结
本文首发于公众号“AntDream”,详细解析了Kotlin协程中`async`与`await`的原理及其非阻塞特性,并提供了相关面试题及答案。协程作为轻量级线程,由Kotlin运行时库管理,`async`用于启动协程并返回`Deferred`对象,`await`则用于等待该对象完成并获取结果。文章还探讨了协程与传统线程的区别,并展示了如何取消协程任务及正确释放资源。
25 0