【Android 内存优化】Android 原生 API 图片压缩原理 ( Bitmap_compress 方法解析 | Skia 二维图形库 | libjpeg 函数库 | libpng 函数库 )

简介: 【Android 内存优化】Android 原生 API 图片压缩原理 ( Bitmap_compress 方法解析 | Skia 二维图形库 | libjpeg 函数库 | libpng 函数库 )

文章目录

一、 图片质量压缩方法

二、 Skia 二维图形库

三、 libjpeg、libpng 函数库引入



在博客 【Android 内存优化】图片文件压缩 ( Android 原生 API 提供的图片压缩功能能 | 图片质量压缩 | 图片尺寸压缩 ) 简要介绍了 图片文件压缩格式 , 以及 Android 提供的图片质量 , 尺寸压缩原生 API ;



在博客 【Android 内存优化】Android 原生 API 图片压缩代码示例 ( PNG 格式压缩 | JPEG 格式压缩 | WEBP 格式压缩 | 动态权限申请 | Android10 存储策略 ) 主要使用了上述 Android 原生 API 压缩图片功能进行图片压缩 ;



在博客 【Android 内存优化】Android 原生 API 图片压缩原理 ( 图片质量压缩方法 | 查找 Java 源码中的 native 方法对应的 C++ 源码 ) 中主要查找 Bitmap.java 对应的 Native 层的 C++ 类 Bitmap.cpp 源码文件 , 并分析了其动态注册 Native 方法的过程 ;



本博客中将分析 Bitmap.cpp 中的源码 ;






一、 图片质量压缩方法


Java 对应方法 :



参数分析 :


long nativeBitmap 参数 : Native 层的 Bitmap 指针 ;
int format 参数 : 压缩格式格式 ;
int quality 参数 : 压缩质量 ;
OutputStream stream 参数 : 输出流 ;
byte[] tempStorage 参数 : 暂时的存储区 ;
private static native boolean nativeCompress(long nativeBitmap, int format,
                                        int quality, OutputStream stream,
                                        byte[] tempStorage);



C++ 对应方法 :


参数分析 :


jlong bitmapHandle 参数 : Native 层的 Bitmap 指针 ;
jint format 参数 : 压缩格式格式 ;
jint quality 参数 : 压缩质量 ;
jobject jstream 参数 : 输出流 ;
jbyteArray jstorage 参数 : 暂时的存储区 ;
static jboolean Bitmap_compress(JNIEnv* env, jobject clazz, jlong bitmapHandle,
                                jint format, jint quality,
                                jobject jstream, jbyteArray jstorage) {
  // 获取 Bitmap 指针
    SkBitmap* bitmap = reinterpret_cast<SkBitmap*>(bitmapHandle);
    SkImageEncoder::Type fm;
  // 判断图片压缩格式 , 给 SkImageEncoder::Type fm 局部变量赋值
    switch (format) {
    // JPEG 格式
    case kJPEG_JavaEncodeFormat:
        fm = SkImageEncoder::kJPEG_Type;
        break;
  // PNG 格式
    case kPNG_JavaEncodeFormat:
        fm = SkImageEncoder::kPNG_Type;
        break;
  // WEBP 格式
    case kWEBP_JavaEncodeFormat:
        fm = SkImageEncoder::kWEBP_Type;
        break;
  // 如果传入未知格式 , 直接返回 错误信息 
    default:
        return JNI_FALSE;
    }
    bool success = false;
    if (NULL != bitmap) {
        SkAutoLockPixels alp(*bitmap);
        if (NULL == bitmap->getPixels()) {
            return JNI_FALSE;
        }
        SkWStream* strm = CreateJavaOutputStreamAdaptor(env, jstream, jstorage);
        if (NULL == strm) {
            return JNI_FALSE;
        }
  // 创建图片编码器 , 需要根据传入的 fm 编码类型创建
        SkImageEncoder* encoder = SkImageEncoder::Create(fm);
        if (NULL != encoder) {
          // ★ 这是图片压缩的核心方法
            success = encoder->encodeStream(strm, *bitmap, quality);
            delete encoder;
        }
        delete strm;
    }
    return success ? JNI_TRUE : JNI_FALSE;
}



源码位置 \frameworks\base\core\jni\android\graphics\Bitmap.cpp


上述 Bitmap.cpp 中的 Bitmap_compress 方法中 , 最终调用的 SkImageEncoder 的 encodeStream 方法 ;


SkImageEncoder 不是最终调用的类 , 而是根据不同的图片压缩格式 , 调用对应的类 , 如果最终压缩格式是 JPEG 格式 , 那么就会调用 SkJPEGImageEncoder 方法 ,



在下面的 SkImageEncoder.h 中声明了 SkImageEncoder 类 , 特别注意下面定义的 virtual bool onEncode 方法 , 是虚函数 , 需要在子类中实现该函数 ;


#ifndef SkImageEncoder_DEFINED
#define SkImageEncoder_DEFINED
#include "SkTypes.h"
#include "SkTRegistry.h"
class SkBitmap;
class SkData;
class SkWStream;
class SkImageEncoder {
public:
    enum Type {
        kUnknown_Type,
        kBMP_Type,
        kGIF_Type,
        kICO_Type,
        kJPEG_Type,
        kPNG_Type,
        kWBMP_Type,
        kWEBP_Type,
        kKTX_Type,
    };
    static SkImageEncoder* Create(Type);
    virtual ~SkImageEncoder();
    /*  Quality ranges from 0..100 */
    enum {
        kDefaultQuality = 80
    };
    SkData* encodeData(const SkBitmap&, int quality);
    bool encodeFile(const char file[], const SkBitmap& bm, int quality);
    bool encodeStream(SkWStream* stream, const SkBitmap& bm, int quality);
    static SkData* EncodeData(const SkBitmap&, Type, int quality);
    static bool EncodeFile(const char file[], const SkBitmap&, Type,
                           int quality);
    static bool EncodeStream(SkWStream*, const SkBitmap&, Type,
                           int quality);
protected:
  // 特别注意 : 该函数是个虚函数 , 需要在子类实现中实现该方法
    virtual bool onEncode(SkWStream* stream, const SkBitmap& bm, int quality) = 0;
};



源码位置 \external\skia\include\core\SkImageEncoder.h



在 SkImageEncoder.cpp 中实现了上述方法 , 其中压缩文件的方法 SkImageEncoder::encodeStream , 在该方法中调用了 onEncode 方法 , 该函数是虚函数 , 需要在子类冲实现 ;


#include "SkImageEncoder.h"
#include "SkBitmap.h"
#include "SkStream.h"
#include "SkTemplates.h"
SkImageEncoder::~SkImageEncoder() {}
// 在该方法中调用了 onEncode 虚函数方法 , 该方法需要在子类中实现 
bool SkImageEncoder::encodeStream(SkWStream* stream, const SkBitmap& bm,
                                  int quality) {
    quality = SkMin32(100, SkMax32(0, quality));
    return this->onEncode(stream, bm, quality);
}
// ... 省略部分代码


源码位置 \external\skia\src\images\SkImageEncoder.cpp



下面的 SkJPEGImageEncoder 类是 SkImageEncoder 子类 , 该类主要处理 JPEG 格式编码操作 ; 在重写的 onEncode 方法中 , 主要使用 libjpeg 函数库实现 JPEG 图像编码 ;


// SkJPEGImageEncoder 是 SkImageEncoder 子类 , 共有继承 
class SkJPEGImageEncoder : public SkImageEncoder {
protected:
  // 该方法中调用了大量 libjpeg 库的函数
    virtual bool onEncode(SkWStream* stream, const SkBitmap& bm, int quality) {
#ifdef TIME_ENCODE
        SkAutoTime atm("JPEG Encode");
#endif
        SkAutoLockPixels alp(bm);
        if (NULL == bm.getPixels()) {
            return false;
        }
        jpeg_compress_struct    cinfo;
        skjpeg_error_mgr        sk_err;
        skjpeg_destination_mgr  sk_wstream(stream);
        // allocate these before set call setjmp
        SkAutoMalloc    oneRow;
        SkAutoLockColors ctLocker;
        cinfo.err = jpeg_std_error(&sk_err);
        sk_err.error_exit = skjpeg_error_exit;
        if (setjmp(sk_err.fJmpBuf)) {
            return false;
        }
        // Keep after setjmp or mark volatile.
        const WriteScanline writer = ChooseWriter(bm);
        if (NULL == writer) {
            return false;
        }
        jpeg_create_compress(&cinfo);
        cinfo.dest = &sk_wstream;
        cinfo.image_width = bm.width();
        cinfo.image_height = bm.height();
        cinfo.input_components = 3;
#ifdef WE_CONVERT_TO_YUV
        cinfo.in_color_space = JCS_YCbCr;
#else
        cinfo.in_color_space = JCS_RGB;
#endif
        cinfo.input_gamma = 1;
        jpeg_set_defaults(&cinfo);
        jpeg_set_quality(&cinfo, quality, TRUE /* limit to baseline-JPEG values */);
#ifdef DCT_IFAST_SUPPORTED
        cinfo.dct_method = JDCT_IFAST;
#endif
        jpeg_start_compress(&cinfo, TRUE);
        const int       width = bm.width();
        uint8_t*        oneRowP = (uint8_t*)oneRow.reset(width * 3);
        const SkPMColor* colors = ctLocker.lockColors(bm);
        const void*      srcRow = bm.getPixels();
        while (cinfo.next_scanline < cinfo.image_height) {
            JSAMPROW row_pointer[1];    /* pointer to JSAMPLE row[s] */
            writer(oneRowP, srcRow, width, colors);
            row_pointer[0] = oneRowP;
            (void) jpeg_write_scanlines(&cinfo, row_pointer, 1);
            srcRow = (const void*)((const char*)srcRow + bm.rowBytes());
        }
        jpeg_finish_compress(&cinfo);
        jpeg_destroy_compress(&cinfo);
        return true;
    }
};



源码位置 \external\skia\src\images\SkImageDecoder_libjpeg.cpp






二、 Skia 二维图形库


Skia 是 C++ 开源二维图形库 , 用于操作二维图形 , 提供一系列 2D 图形处理 API , 在 Chrom 浏览器 , 安卓手机 , 狐火浏览器中使用该图形库作为二维图形引擎 ;



Skia 相关网址 :


官方网站 , 国内无法访问 ;

源码地址 , 国内无法访问 ;

GitHub 源码镜像





三、 libjpeg、libpng 函数库引入


libjpeg、libpng 函数库引入 : Android 中的 Bitmap 就使用到了 Skia 引擎 , Android 中的 Skia 功能不全 , 经过删减了 ;


处理 JPEG 格式图像基于 libjpeg 函数库 ;


处理 PNG 格式图形基于 libpng 函数库 ;


目录
相关文章
|
4天前
|
Linux 编译器 Android开发
FFmpeg开发笔记(九)Linux交叉编译Android的x265库
在Linux环境下,本文指导如何交叉编译x265的so库以适应Android。首先,需安装cmake和下载android-ndk-r21e。接着,下载x265源码,修改crosscompile.cmake的编译器设置。配置x265源码,使用指定的NDK路径,并在配置界面修改相关选项。随后,修改编译规则,编译并安装x265,调整pc描述文件并更新PKG_CONFIG_PATH。最后,修改FFmpeg配置脚本启用x265支持,编译安装FFmpeg,将生成的so文件导入Android工程,调整gradle配置以确保顺利运行。
24 1
FFmpeg开发笔记(九)Linux交叉编译Android的x265库
|
1月前
|
编解码 算法 Java
构建高效的Android应用:内存优化策略详解
随着智能手机在日常生活和工作中的普及,用户对移动应用的性能要求越来越高。特别是对于Android开发者来说,理解并实践内存优化是提升应用程序性能的关键步骤。本文将深入探讨针对Android平台的内存管理机制,并提供一系列实用的内存优化技巧,以帮助开发者减少内存消耗,避免常见的内存泄漏问题,并确保应用的流畅运行。
|
22小时前
|
安全 Java API
Spring工厂API与原理
Spring工厂API与原理
22 10
|
2天前
|
网络协议 Shell Android开发
Android 深入学习ADB调试原理(1)
Android 深入学习ADB调试原理(1)
17 1
|
2天前
|
存储 Java Android开发
Android系统 设置第三方应用为默认Launcher实现和原理分析
Android系统 设置第三方应用为默认Launcher实现和原理分析
16 0
|
2天前
|
存储 Java Linux
Android系统获取event事件回调等几种实现和原理分析
Android系统获取event事件回调等几种实现和原理分析
20 0
|
7天前
|
移动开发 Android开发 开发者
构建高效Android应用:采用Kotlin进行内存优化的策略
【4月更文挑战第18天】 在移动开发领域,性能优化一直是开发者关注的焦点。特别是对于Android应用而言,由于设备和版本的多样性,确保应用流畅运行且占用资源少是一大挑战。本文将探讨使用Kotlin语言开发Android应用时,如何通过内存优化来提升应用性能。我们将从减少不必要的对象创建、合理使用数据结构、避免内存泄漏等方面入手,提供实用的代码示例和最佳实践,帮助开发者构建更加高效的Android应用。
11 0
|
9天前
|
缓存 移动开发 Java
构建高效的Android应用:内存优化策略
【4月更文挑战第16天】 在移动开发领域,尤其是针对资源有限的Android设备,内存优化是提升应用性能和用户体验的关键因素。本文将深入探讨Android应用的内存管理机制,分析常见的内存泄漏问题,并提出一系列实用的内存优化技巧。通过这些策略的实施,开发者可以显著减少应用的内存占用,避免不必要的后台服务,以及提高垃圾回收效率,从而延长设备的电池寿命并确保应用的流畅运行。
|
1月前
|
缓存 移动开发 Java
构建高效Android应用:内存优化实战指南
在移动开发领域,性能优化是提升用户体验的关键因素之一。特别是对于Android应用而言,由于设备和版本的多样性,内存管理成为开发者面临的一大挑战。本文将深入探讨Android内存优化的策略和技术,包括内存泄漏的诊断与解决、合理的数据结构选择、以及有效的资源释放机制。通过实际案例分析,我们旨在为开发者提供一套实用的内存优化工具和方法,以构建更加流畅和高效的Android应用。
|
1月前
|
监控 Java Android开发
构建高效Android应用:从内存管理到性能优化
【2月更文挑战第30天】 在移动开发领域,打造一个流畅且响应迅速的Android应用是每个开发者追求的目标。本文将深入探讨如何通过有效的内存管理和细致的性能调优来提升应用效率。我们将从分析内存泄露的根本原因出发,讨论垃圾回收机制,并探索多种内存优化策略。接着,文中将介绍多线程编程的最佳实践和UI渲染的关键技巧。最后,我们将通过一系列实用的性能测试工具和方法,帮助开发者监控、定位并解决性能瓶颈。这些技术的综合运用,将指导读者构建出更快速、更稳定、用户体验更佳的Android应用。